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Programming/Rails 


通过 此书，你将 学习： 

希望你的网络应 ffl 超越平庸进人 Web2.0 时代？《深入浅出 Rails》 将使你的编 
程和生产力达到最大值。你将学习一 WRiiils scaffolding 的基本原理，以创逮 
自定义的交互式网络应•用程序，全部使 WRails 的-套丰富的工具和 MVC 框 
架。 

你将常捤数据库交 5. Ajax 和 XML 的 Jfc 成.丰富的内容，甚至数据的动态图 
形——曾经要使用 Java, PHP. ASP.NETi&Peri 建立相同的应用程序。你甚至 
可以舒适汴熟练地使闬 Ruby —一 但你是在 Web 编程的上下文中去做这些，而 
不是另一个无聊的 "Hello, World! ”。 



本书的特别之处 在子: 


我们认为你的时间如此宝费以至千不应该彳 t 费在力新概念伤脑筋上而。《深 
入浅出 Rails》 用最新的认知科学和学习理论打造多感宵的学习体验，运用 
适合大脑工作方式的直观的格式编徘，而不是令人&昏欲睡的密密麻麻的文 
字。 
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“《深入浅出 Rails》 延续传 
统的‘深入浅出’系列书 
籍的风格，提供有用的、 
现实世界的信息让你快速 
学习。《深入浅出 Rails》 
对 Rails 学习者来说是一本 
绝妙的书，同样绝妙的还 
有那些最新功能的重温。” 

- -Jnrn/y Durham, 

VVM 开发者 

“我真希望我开始学习 Rails 
的时候这本书已经出版了。 
那样的话我将受益良多。” 

- Mike Isman, 

开发者 



此简体中文版仅限于在中华人民共和国境内（但不允许在中国香港.澳门特别行政区和中国台湾地区）销售发行 

This Authorized Edition for sale only in the territory of People's Republic of China (excluding Hong Kong, Macao and Taiwan) 
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如果布一本兵子 Rails 缟 II 的书，它 
不是只有一大堆理论和的粕车的例 
子， 邡不是很美妙吗？ （ SS 怕达 H 
能躉_个幻想…… 


David Griffiths 著 
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对《深入浅出 Rails 》 的赞誉 


“《深人浅出 Rails 》 延续了深人浅出系列的传统.提供 f 现实世界的有用信息让你快速学习。《深人 
浅出 Rails 》 对 Rails 学者来说是一本绝妙的书， 猢样 绝妙的还有对那些 fe 新功能的1温。” 

Jeremy Durham , Web 开发者 

“與希哿我开始卞习 Rails 的时候这本书巳经出版了，那样的话我将受益良多。” 

Mike Isman , Web 开发者 

“我太爱‘深人浅出 ， 系列了。它们融教疗性和娱乐性为一体！！ ” 

LuAnn Mazza 


“《深入浅出 Rails 》 是对 Web 2.0 迭代开发的一次绝 fM 的广泛介绍。本书将为你展示 Rails 是如何方便快 
拢地开发&饨的 下一代 网站的^ ” 

Matt Proud , 系统管理员和开发人员 

“《深人浅出 Rails 》 是•本我希锘在开始学习 Rails 时便拥冇的书。它以•种幽默而乂字字珠玑的方式 
让你学会你所需要掌捤的基础知 W 。 ” 

Eamon Walshe . 敏捷开发讲解员 



对《深入浅出 Ajax 》 的赞誉 

“Ajax 不仅仅是利用现有技术，对 Web 应用做一些小改动， 然话宣 告它是支持 Ajax 的。 Rebecca 
M.Riordanit- 《深入浅出 Ajax》 中带领你一步步完:成搭還 Ajax 应用的佥过程~，为你展示出 Ajax 不仅 
是‘小小的异步组成部分’，而且也是 Web 设计的更好方法。” 

— Anthony T. Holdenenlll, 《Ajax: The Definitive Guide))" 的作者 


“你不光垃在阅读‘深人浅出’系列书籍，你更垃在实践‘深入浅出”系列书搔。而这就良 1 深入浅 
出’系列书籍 m 类拔举的地方。” 

Pauline McNamara, 大学电子教学项目的技术顾问，瑞士 


“作者以绝妙的 f •法教沣谈者弓 Ajax 衧乂•的各令方面，不仅切中要宵，史以启发读荇，11:读者发 
现 H 题的方式来带领谈者探索有关 Ajax 的各种 M 题。 fr: •呰 还没 fj： 明确笞案的地方，电以开放的观 
点绐读各提供所有选项，并且鼓励读者浊立思考、 f 彳行决策-。” 

Elaine Nelson, 网站设计师 


…处 PAjax 闹境屮？通过阅读冬书，你将彻底 走出 Ajax 的迷茗，你的心智将完全融人 Ajax 的疚心槪念 
并 fi 在幣个过程中体验到寓教 f 乐的 if?IU ” 


Bear BibeaulK Web 应用架构师 




对其他 '深 入浅出”系列书籍的赞誉 


“我昨天 才拿 到这本书并开始读它……然后我就停不下来 J -。 它 的 很酷。 很钉趣的同时， 它 的内容 
也很丰富，而 a 说到 r 点+上》我真的留下了很深的印象。” 

Erich Gamma, IBM 杰出工程师，也是 《Design Patterns 》 的共同作者 

“我成过的关于软件设计的书中最好玩也极 ' W 害的书之一•” 

Aaron LaBerge, 技术副总， ESPN.com 

“曾经漫长又错误百出的学习过程现在被利落地缩简为•本引人入胜的小册子。” 

Mike Davidson ， CEO, Newsvine ， Inc. 

“每一章的核心邢有优雅的设计，毎•个槪念的表达都伴随笤等 M : 的实用性和机智^ ” 

Ken Goldstein, Executive Vice President, Disney Online 


“我爱《深人浅出 HTMU - jCSS 、 XHTML )) 它以种冇趣的形式教你学习你所需要知 it 的所有东 
西。” 


Sally Applin, UI 设计师及艺术工作者 


“平时阅读关 T 设计模式的书或文章时，我小得4、偶尔用什么东西出-压我的眼睛，来确保我的注意 
力集中。但是读这本书时就-点也不需要这样做。也 IT •听起来有点奇怪，但迠这本书确文 il : 学习设 
计模式变得很好玩。 

“当其他关于设计模式的书喊着 ‘Buehler …… Buehler …… Buehler ……’的时候，这本书却在花车上 
商喊•动起来，宝贝!’ ” 

- Eric Wuehler 


简单地说，我爱这本 B 。 实际上，我 At 我套子的面吻了它。 


Satish Kumar 





O’Reilly Media , Inc . 介绍 


O’ReiUy Media ， Inc •是肚界上在 UNIX 、 X 、 Internet 和其他开放系统阁柃领域具有 
领铎地位的出版公司，同时足联机出版的 先锋。 

从最杨销的 《The Whole Internet User’s Guide & Catalog 》 (被纽约公共图书馆评为 
二十世纪最1要的50本书 之一） 到 GNN (最早的 Internet 门户和商业 M 站）， 再到 
WebSite (第一个桌面 PC 的 Web 服务器软件）， O’Reilly Media , Inc . —直处干 Internet 
发展的锒前沿。 

许多书店的反馈表明 ， CTRcilly Media , 〖 nc . 是最稳定的 i | •算机图朽出版商一每一 
本书都•版再版。与大多数 I 十算机图书出版商相比， O’Reilly Media ， Inc . 具有深厚 
的计算机专业背景，这使得 O’Reilly Media , Inc . 形成广一个非常不同干其他出版商 
的出版方针。 O’Reilly Media , Inc . 所有的编辑人员以前都是程序员，或者是顶尖级 
的技术专家。 O’Reilly Media , Inc . 还有许多固定的作者群体一他们本身是相关领 
域的技术专家、咨询专家，而现在编写著作， O’Reilly Media ， Inc . 依靠他们及时地 
推出图书。因为 O’Reilly Media , Inc . 紧密地与计算机业界联系着，所以 O’Reilly 
Media , Inc . 知道市场上真正需要什么图书。 



谨以此书献给 Dawn , 并纪念我的母亲 ， Joan Beryl Griffiths . 



作者 

《深入浅出 Rails 》 的作者 


David Griffiths 在 12岁那年 •§. 到-篇有关 Seymour Papert 
工作的文隹之后侦幵始编程了。15岁的时候，他编写代码 
实现丫 Papert 的 U •算机 iSi ^ LOGO 。 在大学学习 I * 纯数学 
之后，他开始给计算叽? J 代码， Hi 给杂志写文章。他现 
在在英国做& 拢开发 的培训人员，帮助人 ( fl 创达电加简 
电也史 有价值的软件。他的业余时 M 主要花在和 " T 爱的 
者 Dawn —起旅行上， 
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如何使用本书 

介绍 



我炙不敢相信他们拕 
达些东 西故到 了一本讲 
Rails 的书 1。 


在本爷 f . 我 fl 将®普一个辣孕的闷运 
..衿斜么他们把 ii 样的内窖诠夺5 一本 

4%吭， 



如何使用本书 

淮适含阅读这本书？ 

如来对 F 下向所有的问题你都冋答“ 迠”： 

A 你会对 HTML 感到紧张叫？ 

0 你是否 H 冇•门计兗机语言的相关经验，比如 
Java、C# 或者 PHP? 

0 你迫&想意花时 M ftWeb 上搭让些比较酷的东沔？ 


那么这本污鱿适合你， 


谁应泫迗离这本书？ 

如果你对下面任意•个问題回答“ 是”： 


0 你是否从来都没有接触过 HTML? ^_ 

O 你是杏是一名熟练的 Rails 开发人员，正在寻找一本参 
考书? 

0 你是否宵怕尝 W +冏的嚷物？宁可接受牙根管治疗也 
不愿意混搭条纹与花格 f 布的衣服？认为将服务器弓客 
户端拟人化的技术性书径娃不够认真严肃的？ 


那么这本书 就不适 合你。 



£菜 t 丰泳部 f ) 的补充. 
子分句夯 fl 用卡的人] 


>i 本氺这合 


釦 Hii 神作况的锋 . 用不赛找 
心鴂选择由 eLUfl bctW Frtevw.a^ 
知 eric Freest* % 《•• 呆入 

(戋 dHTML 耷 CSS 、 XhfTML 》 . 
烊后爲©的本丰 
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介绍 


我们釦道你在想什么 


“这怎么 " r 能足•本正经的 Rails 朽籍？ 
“这一堆图片都迠什么呀？” 


“我 K 的吋以通过这种方式从中学到知识？” 

我们也知道你的大艋在想 f 十么 

你的大脑说切新祥玩息儿。它总坫：?.找，+1描.等待一呰不 N . V •常 的东沔 
你的大 帖生来 如此，正是 W 为这样的特®， I 能 W 助你生存卩去。 

那么你的大 IR 会如何对待邢呰你毎大 ifti 对的.•成不变. f . 淡 圮 奇的丰 
怙呢？它会尽 ap_liih 这叫氺怙去 T 扰 iftIF . 的 T.fl •: ki * 要紧的亊怙。 

人 W 不会保存那些尤聊的饵怵，它们允法通过•这 W . 然 + S 4 T 的过 
滤机制。 1 

你的大脑如何知进什么是重耍的事情呢?假设有一天你外出远足，然 
后一只老虎跳到了你的面前,你的大 脑和身 体会怎么反应呢? 

神经紧绷.情绪激动，肾 t 睐素激增。 

这躭是你的大®知道的方式…… 

这很重要！请不要忘记它！ 


- 




m 足.想象 t 你 々在家 中.或者在 m 朽馆。这是-个安全、温哚、没饤老 
虎出没的地方。你 iK 在分习、为考忒做准备或荇忒阁学习一叫你的老板认 
为需要花上一周.锻多十天就能学会的商 深 技术。 \ 

何一个 问题。你的大脑 m 帮你一个大忙.它会 w 阁确保这件显然不重 
耍的唭情不会挤占紧缺资源。毕£，资源似好被 ni 来作 fiir «£ 棗 
要的事情，比如老虎、火灾,比如婊近三季«美獨偶《»的胜利 
者。 輔氧也 不弇有什么简单的方法来告诉你的 大脑: “嘿，人脑 
呀，非常感谢你，疋论这本朽 ff 多无趣.多么 lh 我作 作欲睡 ，还 
是済你把这叫东®记下来。” 


你现在的位置 ► 
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如何使用本书 


1 




我们把 u Head Fir si" 的诿考当作 ^£ 


料 ㈣ 必糊新糾 呢‘，肢，你必规了解它.然翩冰•嶋你不会忘 i 己它。这不&_ 
:认 ㈣ 学 ■ 神触辦-_心_的 ㈣ 研究，学仙獅獅 

的柳仅足纸 I : 的文卞 。我们知迫什么耐以比你的大 W 开机。 

Head First 学习守则： 

让它可 视化。 围片比纯文卞 .91 容•.人 W 化 • tfli H •紐-学’】电 A ’* 忖' 
(在知 iU 的 M 想和跳 hfc 多俩_效報升）。它 U ./’ M 内名料 
理解 • 把文字放細 x ； 的 iy 片藝 成片酬 ，而不是 - iK 的底減汽 
P -51， 这样初学者作解读相 - X ； 内容 B . t 达到唞 t - 功佔的效恥 - 

使用对话式和拟人化风格。 W 新的研究衣明， 如* 内容 
演讲#使用第 -.. 人你 • 对话式^•格而小迠过1.£式的 
语气来 a 接与读 t 对话，学屯在课培测甙里的衣现会「 

冇&多 40%的賴。讲述故做报 ft — 。_«式的 H 付 pj 
太严 I 哪-种脈你! J £ 抑染中汗.意力：&晚讀•侣的 K 边细 1 *5, ^ —— 

是-场演讲？ 

, K ⑽学者更深入地思考。换•种说法，除祕积酬动你的神削 fl 胞.__头帖 

f 础结论. to 后形成新知识。而要做到这一点，你宙要接叹•挑战，勘做练 
S ], 以_附思考，_动话化左右_触发多 ㈣ 知。 

引起 读糊 鮮力. 棚肺“抓■科•針，岐賴秘全魏注 
千毎一页 Jr -" 的经验 • 你的大 W 会注息恥哼不洋 
常的.感兴■趣的.怿 异的，引人汴 目的.盘想+ 

到的事情。学习一个全新的.困难的技术课题汴 
不一定葸味笤枯燥•如果它可以变得仃®起来 ， 

你的大晌就会学徘快 W 多。 

调动他細情绪=我们现在知祕的记 忆能膽 大程社_丨猶。 

挪心摘感时，你讀 住它. 我 m 不絲眺》卿_小=====^； 
u fri a »5你解开了-个賴，学会了 it - 他九认为很难的細 • 或者意山、到自匕比肖席了 

m . m . ^…… r 这类情绪， 以及獅 

害！”这样的感觉。 
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介绍 


无汄知 (metacogHitioh ) :想想你如何思考 

如果你真的想学习，而且你希望学习得更快、更深入，请观察你如何集中注 
意力。想想你如何思考，学学如何学习。 

在我们的成长过程中，大多数人并没有学过元认知或者学习理论的课程。我 
们被期待去学习， 但是很 少被教导如何学习。 



0 



我们假定3你拿右•这本朽的时候，你真的希望掌振 Rails ， rfbfl 你可能也不 
希望花费太多的时间。 如果你 希望使用你从本书中读到的知识，你需 要记住 
你所读到的内容。而要做到这一点，你就需要理解读到的 内容。 为了从这本 
书或者 K 他书和学4经验里获得更多的知识，你需要对你的大 W 负责，让你 
的大脑关注这些内容。 


秘決是让你的大脑认为你正在学习的新学的知 i 只确实很重要，与你的生 
死存亡有关，就像吃人的老虎一样。否则，你就会陷人一场长期战役 
里，你的大脑依然会尽它娀大的努力来使新东西无法留下痕迹。 


所以问题在于你如何让你的大脑把 Rails 当成一只饥饿的老虎 
来看待？ 


有缓慢且繁琐的方法，也有 快速且 有效的方法。缓慢的方法就是纯 
粹的重复。你当然知道再乏味的知识只要你反复学习总能学会和记住。只要重 M 
的次数足够多，你的大脑会 觉得： “虽然 这似乎 对他小 那么® 要，但是他翻來覆去地 
看这个部分，所以我觉得它应该是重要的。” 


而快速的方法就是做增强大脑活动的事，尤其是不同类型的大脑活动。前•页提到的 
就是解决方案的大部分内容，它们都被证明可以用你所@坎的方式来帮助你的人脑工 
作。例如，研究表明把文字放到它所描述的图片里面（而不是置子®面内其他地方， 
如 m 片说明或正文）会让你的大脑尝试理解文字和图片的关系，而这会触发更多的神 
经元被激活。电多激活的神经元就等于让你的大脑有吏多机会来知道这是一些值得注 
意的亊情并且记录下来。 

对话式风格的帮助是由干当人们感知到他们正在一个对话中时，他们会试图加强注意 
力，因为他们必须竖起耳朵，注意整个对话的进行。更让人惊書的是，你的大脑并不 
在意这个■•对话”是发生在你和书本之间！另一方面，如果写作风格是正式而枯燥的 
话，你的大脑觉得它和你正在聆听一场讲演，自己只是满屋子的被动听众中的一员， 
没有必要保持清醒。 

但是图片和对话式风格还只是开始…… 
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迖 I 我们所 傲的： 

我惘使用图 片1 因为你的大肭对视觉化效果较有感觉，而不是文宇。只要能够引起 
你的大脑的关注，一幅图片胜过千言万语。当文字和图片一起出现时，我们把文字 
嵌入在图片中，因为当文字放置于它所涉及的图片中，而不是图片说昕或埋没在正 
文中时，大脑会运作得更有效串。 

我们重兌表现柑同内容，把一件事情用不同的方式和4、'同的媒介以及多种感知来增 


AjaxiJI 求 
HTML^g _ 

■ 丨丨 "J Ir* I 


加内容被记录到你的大脑的多个区域的机会， 

我们以意想不到的方式:来使用溉念和图片，因为你的大帖更容易接受新奇的事物，而我 

们使用的图片和构带有情感内容，因为你的大脑吏容易关注生理情绪。这会导致让你1 二 j 7\\ [ 
有感觉的事物更容易被记住，即使这种感觉只是一点点幽默、惊讶或者有趣。 

我们使所拟人化、对话式的 K 格，因为如裝你的大脑次为你处于一个对话而不是被动地 


听演讲时会更容易集中椅办。即使你在阅读时，你的大脑也是这样工作的。 


我们弓卜人了超过80个练习活动^因■为当你在做半情而不是读事情时你的大脑会学习和记 
住更多的内容。而我们让练习维持虽:然有战性但是依然能做到，因为这是大多数人所 
期望的。 

我们使用多种学习风格，因为你可能喜欢循序渐进的过程，有些人可能希望首先了解 
整体轮廊.而其他一些人則希望看到一个例子。但是无论你倾向千哪种学习方式，每 
个人都可以通过多种漁度来看到相同的内容，从而受益。 



我们为你的左右肭都准备 f 内象 因为你让你的人脑参与得越多，你就越有吋能学会并 
记住，以及保持更长时间的关注 d 于让一 边:大 swim 通常意味 笞吐另 一边有机会休 
息，你可以学习得吏久、史有效串。 

我们加入了提供不同秘点的故事和练习，因为你的_大脑在它被迫作出评估和决定时能够 
学习得更深入。 


< 5 ^ 


书中也有相当多的挑战和练习，通过问问题的方式进行，答案不见得都彳 K 直接，我们的 
用意是让你的大脑深涉其中•-学得£多，记得更牢。想想看一一你不可能只是通过观察 
健身中心的其他人来让自己塑身。但是我们尽我们的最大努力来确保当你刻^?学习时， 
你所学的是正确的 事情。 也就是你并没有浪费额外的脑力去处理•一个难以理解的例 f , 
或者分析_的，冇着太多术语的或是过 i ； •简单的友字。 

我们使用人物 • 在各种故事二例子、图片电使用人物，因为 r 嗯，因为你是一个人^你 
的大关注干人而小是其他事物^ 
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7 t 你的大胳顺从你的方法 

那么，我们完成了我们的 部分。 剩 K 的就1〗你了。这些技 
巧只是一个开始，请聆听你的大脑，找出哪呰适合你，哪 
些不行。试试看吧！ 

: . 3 巧线专下.钻砧到仿的 


O 减慢阅读速度》你理解的越多，你需要强记的就 
越少。 

请小要只&阅读《冇时你需要濘下来进行思考。当 
这本书问你一个 N 趙时.+要直接跳到答案部分。 
想象一下冇人 K 的在 H 你这个 IW 题。你 it 你的大 P 
见考 得越深人，你就越有机会学会和记 ft ® 多知识。 

o 做练习，记笔记。 

我们安排了练习题，但是如果我 ( n 帑你完成练习 
题，那就像让其他人#你训练 -- 样。不要只是看 
练习 题。请使用你的铅《。冇很多证椐表明，在 
学习过程中的实质活动吋以强化学习效果。 

o 阅读'■没有蠢 问题" 单元。 

IIP •细阅读所有的“没有蠢问题”。它们不是无关紧 
要的说明，而是核心内容的一部分。千万不要忽略 
它们！ 

o 让它成为你上床睡觉前最后阅读的内容，或者至少是 
最后一件有挑战性的事情。 

部分学习过程（特别足锯耍转化为长期记忆的情况 
F ) 发生在你放下书本之后。你的大脑也需要一定 
的时间来对学习的内容进行处理。如果你在处理的 
时候放人其他的新鲜事物，你刚刚所学习的部分内 
容就会丢失。 

0谈 论它。 响亮地说出来。 

说话是大脑不同部分的活动。如果你试图理解什么， 
或者想强化记忆，你应该把它大声说出来。而更好 
的做法是，尝试大声地解杼给其他人听。这样你就 
会学得史快，你会发掘出一些你在阅读的时候没有 
想到的新想法。 


Q 喝水。大屋 喝水。 

你的大脑设泡在充沛的液体中时工作状态最佳《 

脱水状态（往往在你感觉干渴前会 发生） 会减 
缓认知能力. 

Q 聆听你的大脑。 

注盘你的大 W 坫 S 过度疲劳。如*你发现你 lL 经 
在浮光掠影地阅读或者开始遗忘你刚刚读到的内 
容时，你就需要休息了。一旦你过了一定的时间 
点，你鱿不吋能通过用强塞内容来加快学习速度， 
而 R 你这样做还吋能破坏学习过程。 

o 用心感受。 

你的大^宙要知道什么是 a 要的事怙。你应该进 
入到故事屮。根据提供的照片展开自己的想象。 
即使因为一个褙糕的笑话而抱怨也比什么都感觉 
不到来得强。 

0练习编写 Rails 应用！ 

想要褚通 Rails 编程只有一 条路： 编写 Rails 应用- 
而这正是你通过这本书所要做的。爷握一门课程 
的敢佳手段就是实践，熟能牛.巧。我们将给你提 
供大 M ； 的 练习： 每一章都会有 ft 要搭违的应用》 
请不要跳过它们一学习的成效躭在你自行搭违这 
些应用的过程中逐步产生。出错也不用担心•事 
实上，你的大脑从错误中学的速度要比从成功中 
吏快， 垴后，请确保你在进入下一章之前已经理 
解本隹的内容=毎一章都逑立在前面茕节的基础 
之上。 
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读我 

这是一本讲述学习经验的书，而不是一本参考书。我们在这本书中特意去掉了那 
些可能会阻碍学习进程的内容。当你第一次阅读的时候，你需要从首页开始，因 
为这本书对你已经看过和学过的知识做了 • •些假设。 

在阅读本书之前，你需要先把 Ruby on Rails 安装到你的机器上。 

这不是一本提供入门知识的书，所以我们不会有任何章节来告诉你如何安装 Ruby 
on Rails 到你的机器上。你最好从 Web 获取相关信总。你需要安装 Ruby on Rails 2.1 
版或者更高版本，同时也要安装 SQLite 3。你可以从如下_址获取更多的 信息： 


http:// www. rubyonrails. or g/down 。 


这不是一本参考书。 

所以不要期望会有大段大段的文字来解释如何用 15 种不间的方法来实现某件事 
情。我们希望你通过实践来掌握知识，所以你可以直接开始。我们会给你足够多 
的信息来推动你的学习。当你到达本书结尾时，你就会对 Rails 是如何工作的以及 
它可以做些什么有•个思维框架 # 然后你就能够比以前更为迅速和有效地把参考 
资料装到你的大脑里。心理学家把这称为组织信息的能力。 

本书的所有代码都可以在 Head First 网站获得。 

我们会在前进的过程中逐渐展示出所有你所需要的代码。与本书一起编程是良 
策，而在完成代码的同时也让它实现一些你自己的想法则吏加绝妙。有时你可能 
想要-份每章中的代码拷贝，所以我们把这些代码放在了 Head First Labs 网站 
上。 Rails 应用具有很强的独立性，所以你完全可以在本书所实现的功能代码基础 
上再精雕细琢一番。你可以从如下网站下载 代码： 


http://www. headfirstlabs. com/books/hf rails 


我们并没有完整地解释每段代码。 

Rails 能够为你生成大段的代码，而我们并不想 It 你陷入逐行的描述里。我们将描 
述你需要知道的重要段落，然后继续向前。不用担心 —— 到本书结束的时候，每 
段代码都会变得清清楚楚的。 
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这是一本讲 Rails 的书，而不是讲 Ruby 的书。 

Ruby 是编写 Rails 框架的语言，在本书的学习过程中，我们将教会你刚好 
够用的 Ruby 知识。不用担心，如果你已经有一些使用其他编程语言的经 
验，比如 C # 或者 Java ， 你会一切触利的。 Rails 是一个功能非常强大的系 
统，你只需要很少的 Ruby 知识就可以驾驭它。 

所有的活动都必须参加。 

练习和活动不 是附属 品，它们是本书核心内容的组成部分。有些帮助你 
记忆，有些帮助你理解，还有些帮助你运用你所学到的知识。请不要略 
过任何练习。 

重复是刻意的，也是重要的。 

“深入浅出”系列书》的一个显著的差异在于我们希望你真的掌提它， 
我们希望你在读完本书时能够记住你所学到的知识。大多数参考书并不 
会把你能够记住多少内容作为它们的目标，但是这本书是讲述学习的， 
所以你会多次看到相同的槪念。 


程序范例尽量精简。 


读者告诉我们，费力地看完10个仅有微小差异的相同代码段的不同版本 
是件非常打击人的事情，所以有时我们仅仅显示脚本的修改过的部分。 


章节是以技能而不是技术为导向的。 


每章都会教给你相应的技能来编写出越来越先进和有价值的应用。所以 
我们不会在某个章节仅仅讲述如何连接数据库或者设计一个漂亮的界 
面相反，每章都会教给你一点点数据库知识、一点点界面知识以及一 
点点 Rails 其他方面的知识。每一章结束后，你就能够说，“酷！现在我 
能够搭逮一个实现 X 的应用。” 
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技术审阅 专家： 

Andrew Bryan 馇来 fi 新西 A 奥克兰的软件开发和商业 
顿问。他 H 前在一家位干波士顿的在线媒体及广告公 
司工作，他与他的舂子 Agie 生活在一起。 

Jeremy Durham 从2005年初就开始使用 Ruby on Rails 
搭建 Web 应用。他与他的妻子和两个孩子生活在马萨 
诸寒州的阿林顿。 

Matt Harrington 是美国东北大学的校友，他从9岁起就 
是一名狂热的程序员。 


Iivingtol 00 .com 的寿命计算器 (Life Expectancy 
Calculator ) 。 他 T 2004 年毕业于罗切斯特大学并获得 
计算机科学的学位，从那以后他便一直从事 Web 开发。 

LuAnn Mazza 是来自干 Hlinois 的计算机分 折师。 

Eamon Walshe 是 Exoftware 的敏捷开发训统师•之 
前是 IONA 科技的杰出工程师。他是一名 Rails 爱好者, 
因为 Rails 能够让开发人员专注于有意义的事情——快 
速实现真正的商业价值。 


Mike Isman 自2006年初加入 eons.com 团队后就开始使 
用 Ruby on Rails , 那时 Rails 1.0 还没有发布。在 Eons 
工作期间， Mike 用 Rails 编写了一些小型网站，包括 
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Rails 把 IE 录转化成对 fe 68 

数据在内存中，而网豇可以#见它们 69 

有个问题 一 用户找不到他们想要的网页 73 

路由按照优先级顺序运行 76 

为了把数据放入视图，你还霜要控制器中的代码 78 

索引豇面需要来 G 所有记录的数据 79 

AdTind (: all) 一次读取整个数据表 80 

数据作为一个称为数组的对象返回 81 

数组就是一些编号后的对象序列 82 

用 for 循环读取所有的广告 86 
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但是有两个页面揆板……我们要修改毎个模板的代码吗？ 94 

但是 MeBay 发送过来的新的静态内容怎么办？ 97 
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槠入> 更新 和删除 

一切都在变化中 

变化无处不在尤其对数据而言. 到 U 前为止，你已经 肴过如 何通过 
支®来迅速槁定一个 Rails 佐用，以及如 M 编〜 J 你 Q 匕的代码来展示数椐库中的 
数据。 W &， 如果你希望用户能够自行编辑数据的话应该怎么做呢?如果支架 


无法实现你所期墮的方式呢？你将会在本章学习到如何用你所希窀的*式来插 
入、更新和删除数据。而当你这么做的时候，你就会更深人地了解 Rails 是如何 
运作的，而辻你还能学到一点儿安全知 m 。 


人们希望在线张贴新广告 104 
你已经知道如何搭建一个发布数据库中数据的应用 105 
保存数据就像读取数据的反向那样工作 106 



你需要•个用宋提交数椐的表单和一个保存此数据的动作方•法107 


表单与对象相关吗？ 109 

Rails 能够创逮与模型对象相关联的表申. 11() 

® ad 表单对象还没有被创逮 114 

表单对象锯要在表单被51示之前创逮好 115 

表单广告对象将在控制器的 new 动作中被创建 116 

现在每个页面模板都有-个匹 gd 的控制器方法 117 

表单不会发送回对象，它发送回数据 119 

Rails 锯要在数据被保存之前把数据转化成对象 120 

手把手教你控制器的 create 方法 121 

控制器需要保存记录 122 

不要创建新页面，使用现杆页面 128 



但是控制器动作如何才能显示另一个动作的页面呢？ 
茧定向 使控制器能够指定显示哪个视囹 
但是如果广告在张贴沿黹要修改该怎么办？ 

史:新广告就像创建广告一样……只有一点小区别 
你需要找到一个广告而不是创建它， 

你需要 E 新这个广告而不是保存它 

限制对某个功能的访 N 

…-但是现在旧广告黹要被刪掉 

自己编写代码4以让你实现比支架更多的功能 
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掩库金询器 

事实还是推论？ 


你作出的每个决定都会有其原因。 ftRail 沖，知道如 M 作出明智的决定可以节 

宵你的时间和精力。 我们 会在本章来看- I、 用户的需求迠如 M 在应用的一开始就影 
响你的选择的。你佐泫使⑴支架然 Vi •洱柊改中.成的代码？ 佐该 ft 接从头开始创达代 
码？ 无 论哪种 A •法，一旦你开始定制 Cl 己的应用，你就必须了解 查询器 （finder) : 
通过•种杆意义的 // 式凇获 取数据 If •山此满足 用户的需求。 
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査询器生成数据库査询 179 
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验证你的数椐 

防止错误 

每个人都会犯错……但是很多错误都可以被避免！ 即使你 的用广•全神 贳注， 
他们依然可能把错误的数据输人到你的 Web 应用中，使得你不得不妥善处理它们。 
但是请设想一 F ， 如果有一种方法可以在第一时间阻止错误的发生会如何呢？这 
就是为什么我们要引人验证器 (validator) c 请继续读下去,我们将为你展示如何 
添加聪明的 Rails 验证器到你的 Web 应用中，使你能够控制什么样的数据被允 i ' r 输 
人以及什么样的数据应泫被徘除在外。 
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我们还要 W . 示错误消息！ 212 
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建交连摟 

把它们集合起来 

团结就是力置。 迄今为 i [:， 你 ll 经知道了 Rails 的- •些关键组成部分。 你创 yrr 
赘个 Web 应闬汴 ii 裉据你的需要对 Rails 产生的内容进行 r 定制。 但在现 实世界 *11 ， 
情况可能更为复杂。 请继续读下去……是时候来创建 j 些多功能的网 页了。而且， 
Hi 足时候来处 洲复杂的数据关系和 通过编 g 你 &己自 定义的 验证器 来拉制你的数据 
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减少 

人们孝 

Rails , 


ax 

少流量 


人们希望获得最佳的生活体验……以及最佳的应用体验。 无论你多么精通 

对干一些传统 Web 应用你可能也会做得不那么成功。有时候，用户想要电 


动态的东两或者能够 N 应他们的紐•次突发奇想的东两。 Ajax 使你能够搭达快速 
的、反应灵敏的 Web 应用，它是专门设计用 来给了 •用户 Web 应用必须提供的最佳 
体验，而 Rails 已经提供 f 一组它自己的 Ajax 库，你随时可以使用。是时候方便快 
拢地把 Ajax 褚华添加到你的 WebA : 用了， ilJH 户享有比以前史体:的体验。 



邴 f 肮空有个新的促销 it •划 264 

页面的哪•部分变化最快？ 265 

浏览器不总是史新整个页面吗？ 270 

还有什么方法 " T 以发送请求？ 271 

首先我们需要包含 Ajax 库…… 272 

. 接 卜 •来我们需要添加 •个 Ajax “ Refresh ” M 接 273 

浏览器 需要乜 动询问更新 278 

我们一定要 ih 浏览器反复洵问吗？ 279 

你可以像监听按钮事件一样监听定时器事件 280 
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有些人的单汸派对有问题 f 285 

表单需 要生成-个 Ajax 请求 286 

表竽需要由 JavaScript 来控制 287 

我们需 要锌换 create 方法 289 

这段代码会有怎柞的效果呢？ 290 

肮班预 U ■有个问题 295 

我们只知道如何一次更新页面的一部分内容 2% 

控制器需要返回 JavaScript 而不是 HTML 297 

那么 Rails 生成 T 些什么？ 301 

如果你没有表明在哪儿放 S 响应，那么它躭会被执行 302 
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XML 和多种表现形式 


现在看起来都不一样了…… 

你不可能一直取悦每个人。你能吗？ 我们已经符过了你如何使 ftlRails 来快 
速月.简单地开发能够完美满足一系列需求的 Web 应用。但是如果又打其他需求出 
现的话该怎么办？如果冇呰人想要基本的网页，而另外一咚人则想要 Google 混搭 
(mashup) ,还冇史多的人希望你的应用能够作为 RSS 源 (feed) 存在，你又该 
怎么做呢？你将 ft 本茕中创建同样某础数据的多种表现形式， i 上你能够以最少的代 
码来？ :.现最大的灵活性。 
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312 
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313 
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314 
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315 
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316 
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318 
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319 
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320 

控制器代码肴起来足什么样呢？ 

321 

此时 • 在20 000英尺的高度…… 
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我们黹要生成 XML 和 HTML 

327 

XML 和 HTML « K 只是衣埂形式而已 

329 

我们应该如何确定使用哪一种格式？ 

330 

地 m 菸面是如何工作的呢？ 

334 

代码吋以正式运行了 

336 

RSS 源就是 XML 

344 

我们将创建一个名叫 news 的动作 

345 

我们必须更改 XML 的结构 

348 

所以我们将使用一种新 模板 ： XML Builder 

349 

埂在让我们把源加人到豇面中 

353 

在世界最高点！ 

355 
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REST 乌 Ajax 

更上一层楼 

是时候巩固你的混搭技 术了。 到I彳前为止，你已经符 过如 何添加 Google 地 
图到你的 Web 应用中宋消晰地展现空间数据。但 是如果 你恕耍扩展已经存在的 
功能呢？ irl 继续该卜‘去，我们会为你展示如 M 添加更先进的 Ajax 精华到你的混 
搭式应用 （mash-ups) 中。而且，你会通过这种方式学习到更多 REST 的知识。 



奉汁太多了！ 358 

地图可以显示更多的详细信怠 359 

我 ff 1能够扩展祐干 Ajax 的地阁 360 

但我们怎样才能改变索引页面呢？ 361 

“ show ” 动作需要生成怎样的内容？ 362 

新的地图功能成功 f ! 367 

我们也诟要使用 Ajax 来创建诂求 368 

地围局部揆板"〖以让我们指 定一个 “ new ” 动作 370 

我们如 Will •:明一个事件已经被保存？ 375 

表单需要史新弹出窗丨1的<加>的内容 376 

雪崩！ 381 

现在编铒是如 M 实现的…… 382 

我们可以在弹出窗口中加人一个 “ Edit ” 链接 383 

我们从修改 “ edit ” 动作开始 384 

我们还需要在 show 页面 h 增加一个新链接 386 

那么我们该如何使用 Unkjo 辅助函数？ 387 

让 Ajax 链接宋•拯救我们 391 

我们 HI 错 f 路山！ 393 

HTTP 方法是选择路由的一个因桌 394 

什么是 HTTP 方法？ 395 

Head First Climbers 黹要你！ 398 
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炱实世界里的应阁 

真实世界里的 Rails 

你已经学到了很多 Ruby on Rails 的知识。 川.足为丫能够把你的知识运用到 
真实世界甩，你还需要思考儿忭唭愔。你矣如 M 把你的应 HI 连接到另一个数据 
库?你该如何测试 Rails 应用?你如何 >1 •能最有效地使用 RaiMllRuby« 言?你可 
以在哪 )L 找到 Rails 的最新进展？ W 继续读 下么，我们会 ih 你山•领行利地形并由此 
把你的汗发技能提升到豇高的 M 次。 



#!这儿有满满-页的 Ruby "试试看” 
Web 应用也溢要测忒 
釕哪呰龙喂的肿 IbMHi 呢？ 

1:线运行 

那么你该如何电改数据庳紀咒 WG 
什么足 REST? 



迷失方向的 Web 应用 
生活在 Edge 上 
获取史多的信息 
消遗性读物…… 

相乂话题的 ** 深入浅出”系列屮;锫 
离开 Rails 村…… 
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1 孖始？ 


命 

本飞驰的 Rails + 



想让你的 Web 应用开发像飞一样快吗？那么你需要了解 Rails 。 

Rails 是现有 最酷、最快速 的开发框架，它可以让你以从未想象过的速 
度开发出具有完备功能的 Web 应用。开始丫解 Rails 很简单，你所需要 
做的只是安装 Rails ， 同时翮过此页继续阅读。在你明白之前，你已经 

领先好几英里了。 


这是新的一章 




欢迎来到星期五 


星期五，罕上 9 点 


你收到的第一封 E - mail 来 ft —个碰到麻烦的老 朋友: 


2杖星前荃的邱扣 
e - wcflll 0 


嗨 tt 女”忙 I 还记得我说过的那个我们正在做的售票 
削 记经为它忙了几个星 期了！ 开发团队确 

实遇到了一些 难题。 

你觉得你能帮我们创建这个应 用吗？ 

牛的网^ w 喊 ㈣ 

( . 条新的门票销售记_ ^ 

i 读取和显示单一的门票 / 

\ . 更新某一销售记录的细$/ 

除一条门 


袪 s 沒这个 
rtf 廬荈餚构 i 
贫现 。 


name ——- 购买者的姓名 ㈣ ing ) 
seatJOeq —— 座位号比如 E14 ( string) 
address ——购买者的地址 ㈨ 叩 string ) 
price . paid 一一门票的销售价格 ( decima ^ 


ail address — 


址 ( string ) 


我还附上了网■雜，这__道飾的目 碱 ■的了。 

哦！ 而且我们周 - 前就需要完成这一切，否则我的屁股放要遭映了. 


这个系统蛙被设计给音乐会演奏厅的前台工作人员使用的。.在每场音乐会 
之前数据库都会淸空,.所以毎次它只需要记录_.场音乐会的具体怡况》你 
觉得你能帮上忙吗？ 
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开始 T 


这个应阁熏荽傲很多攀件 

这儿有-些网页的草图_。它值】是怎样迎合系统需求的呢？ 



你现在的位置> 





了解你的需求 

那么兵子迖个应用我们 t 要些 f 十么嚯? 

为了在演奏厅的服务器上运行这个应用，我们需要几样东 
两。我们 需要： 

o 一套应用框架。 

我们需要-套预先写 4 f 的代码作为这个 Web 应用的基础。 

o 一个数据库系统。 

我们需要把数据在某种数据库中。 

O 一个 Web 服务器。 




Ralls 是怎#帮助我们的哝？ 

不论你用什么语 a 来写代码，你开发的应用很可能仍需要这三 
样东西。 Rails 做的非常好的地方之一就在干它包含了所有这些 

你需要的软件-而且它们都是免费捆绑在里面的。 

让我们来看看这一切是怎么实现的。 
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代碚泜谜拯 



Rails 里内置了很多特性。你需要做的是猜出这 
个代码池里哪些特性是这个 Web 应用^ 
要的，然后把它们写在下面的空行中。你 
并不需要所有这些特性。 


❶ 

o 



注意： 代码池里的每 



样东西都只能使用一 

次！ 


你现在的位置 ► 











数据驱动应用 .. ..... . 

Rii 紐勝那些姑数椐库为讎 
心的皮用，就像这个售累系统 

很多应用的核心部分都弇包括一个数据库。这些应用存在的主要原 
因是为了让.用户可以通 过仓们 来»'|«]她改数据座的内容，而不 
需要#接使川 SQL。 

那么9你把一个数据 H •:连接到-个 Web 应用时，需要解决什么 M 题 
.呢? . 

首先，这个 Web 应甩盂要 t 许用卢访问和史改数据，所以 Railsfci 含 
了一个名为 AdionPack 的应用框架 (application framework) , 
个应用.框架吋 d 帮助你泱数据职动的女互页面。 






代碚泜键拯解著 



Rails 里内置 T 痕 多特性。你 
需要做的是找到代码池里 
的三个特性，它彳 n. 是这. 
个 Web 应用所需要的，并 
ffi 它们写在下面的空行中V 




,你.弁不需要 m 有这些 特性， 



其次， Web 荜用需契运•行在 1 台 W 叫服务:器上 以显示 这孽网坪，所: 

以 RaUs 包条了一台内置的 Wd ) 服务器 6 ■' 

• .... •• . . . 

第三，你耑要一个数据库。 Rails 创达的 Web 应用被预筅配授为龅够 

和，个集成的 5 QLite 3 数抬燦一起工作。 

........ . ■ . . - 

:你龠要的第四个东西是 : 个 巧象关系映射库，为此 Rails 提供: 
了名 ， ActiveRecord 的对象-关系映紂库 6 它可以让 你的数 据库義 
起来像一个简单的 Ruby 对象集今。 

降了这些之外， Rail 纟还 k 括了一堆工具脚本来帮胁忤管理 k 用。 
所以，如果 f 尔正在创建十个以数据库为中心的 Web 应用，你就会发 


ActionPack framework 


Bur>dled HTTP server 


SQLite3 RDBMS 


ActiveRecord library 

I 聯 ; 



Rails ^ 你扶伊 3 佤霈要的所有方西。 


論 tV 
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开始了 


你玎认 iDuili 命令采创建一个 
新的 Web 应用 


那么你如 W 开始使用 Rails 呢？ 

用 Rails 来创迖一个新的 Web 应用真的非常简申.。你所崙要做的 
只是打幵一个命令提示符或者一个终端窗口，然后输入 rails 
tickets, 这里的 “tickets” 就是你想要创逮的 Web/*V: 用的名字。 


r 


+ 

迗样炒 

命 



C. % ^ 命今枝孑符.会鲶 
入 Vails tlotects" « 


它会做些什么哝？ 


tickets 

README 
Rakefile 


输入 rails tickets 之后 Rails 就会聪明地在一个名 
为 “tickets” 的新文件夹里生成一个 Web 应用。还不止 
这些，在这个 tickets 文件夹里， Rails 生 成了 整 套的 
其他文件夹和文件，它们组成了一个新的 Web 应用的 
基础结构。 

这意味若你仅仅通过一个简单的命令就已经成功地创 
违广一个完整的基础 Web 应用。 



故松 


Rails 生成了许多 
文件和文件夹，但 
是别担心。 


它们的存在都是有 
原因的，你会在本书结束的时候押 . 
解它们所做的一切。 


RlOsJ 过一务舍今杖.冷伤 
法成3管套的之蜱扣艾件 
夹： 这泉釔的蝥个 Wtb 
左°用的锌构。 


函 | app 

config 

^ ® db 
im doc 

-^ aa ,ib 

-»ijp '°g 
~ > SHI P u b" c 
— 睡 1 script 

-> 涵 test 
— (m tmp 

~~^ [jfl^ vendor 


你现在的位置 ► 










试驾 



因为你刚刚创逨的应用是一个 Web 应用，你需要启动内贾的 Web 
服务器来看到这个应用运行起来。 


在命令提示符或者终端窗口，切换到 tickets 文件突并输人 “ruby 
script / server ” . 


进入这个&用錡 
ft 的 t 件夹 …… 

7 


File Edit Window He<p 


> cd tickets 

> ruby script/server 


/ 


iiS •- 个糾 H 
M Widows6*) 命令穿 0 
Li^uoc 扣的终揉逬入它。 



. fis ^ web 

吸务器。 


屏幕 i ： 会出现一些信息来表明这个 Web 眼务器正在运行。现在 
你可以通过浏览器打开下面这个地址来看到默认 主页： 
http : / / localhost :3000/ 



器的 i 蚤。 


扪穹私始 I — 

Rails 默认在端口 3000启动它的 Web 服 
务器。如果你想使用别的端口，比如 
8000,那么运行这行 命令： 


ruby script/server -p 8000 
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开始了 


现在 f 要在默汄应用中加入你 f 3 
的代码 

Rails —开始就会创逑你的应用的基本框架，但是你仍然需要根 
据你的 R 体崙求添加代码。每个人的具体应用都足小同的，小过 
Rails 是不是有工具成者捷後来 让你更 简单地生成自定义代码呢？ 

嗯，实际上， Rails 确实有。你有没有注意 Rails 足如何生成-套完 
铍的文件框架的？简直就像它知道你将要 W 到什么似的。这是因 
为 Rails 应用遵循了非常 强人的 命名约定。 

Rails 应用总是遵循约定 

所有的 Rails 应用都遵循 | fi ] 样的堪本文件•结构，使用一致的命名规 
則。这会使得应用电容易被押.解，同时它也怠味苕内 S 的 Rails 工 
具 " r 以理解你的应用是如何工作的。 

为什么这柞 做很重 要呢？哦，那是因为如果这呰工具明 d 了你的 
应用是組何组织的，你就能使用这些工具来自动完成很多编码任 
务。通过这种方式， Rails 能够使用约定来为你生成代码，而不需 
要你来 fidWWcb 应用。也就是说， Rails 遵循约定优于配置原则 
(convention over configuration ) c 


约宏饮 子郎置 


让我们 来瞧瞧 Rails 最强有力的工具之支架 （ scaffolding ) 。 


产没有蠢纖 


问： 


你一直提到 Ruby 和 Rails , 它 


们有什么区 别呢？ 


答： Ruby 是一神编彳1语言 t Rails 是 
一系列 Ruby 脚本的集合„所以 Web 服务 
器. ActionPack 应用愜架 还有那 些捆綁 


的工具脚衣实际上都是 Ruby 脚衣…… 
也就是 Rails 的一部分„ 


问： 


我该怎么编辑新网站的主页呢？ 


通过应用目录下的 public / 


index . html 这个 HTML 文沣》这个 
public 目录包含了应用的所有錚态内 
容。 


问： 


如果我想启用一个不同的 Web 
服务器会怎么样呢？可以这么做吗？ 


在开发过 裎中使 用内芄的服务 
器比较合理。不过如果你希望把应用 
的上线版本发布到另一个 Web 眼务器 
上也是可以的。 


问: 


当我运行 ruby 


script / 


server 时，具体在哪个目录下运 
行有关系吗？ 

是的.确实有关系。你必煩在 
包含 Web 应用的文件夹内运行命令。 

1^)* 谁来编译我的代码呢？ 

答： Ruby 是一种解释语言，和 
JavaScript —蛘_这就意味着它不需要 
编译。你可以直接更改你的代码.并 
立刻运行它。 


你现在的位置 ► 
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生成你的代码 


支架就是生成的代码 

那么我们的&用溢要做哪些事件呢？ it 我们来歌新看 
_ _ _ _ 


ii 杖 基笏面 的邳扣 


确 If I 奄这个 
0#裊轉锘均上 
淀现。 


酶一 — 你好 T 还记得 _ 过 _ 个我祗 _ 的售 J 

SSf 它帮的我 “ 不 A 焱已经为它忙了几个星 期了！ 开发团队确 

' 实 ' 遇到了一些 难题。 

你觉得你能帮我们创建这个应用吗 7 

牛的网 

已售出 |* <u 
f . 条新的 n 票销售记 ^ ㈣ 料 

I 读取和显示单一的门票 \ 

\ . g 某一销售记录的 

但是老板说他们至少需要这些特 
构 如下： 

( name 一购买者的姓名 ( stri 叫） 

f seatjd _ Seq —— 座位号比如曰4 ( 细叩） 

V address——- 购买者的地址 (lon9String) ) 

一 ^ price paid—— 门票的销售价格 (decimal)/ 

address 二购买 止 ( s _) 

^ 班嫌 t 了网页的草图，这样你就知道我们的目标是啥样的了、， 
織轉艇細 _ 藤鮮紐了！ 


所以，我们；创达网 沉来允 ITflUH 创建 （ Create > 、读取 
( Read > 、更新 ( Update ) 和删除 （ Delete ) H W 为这作操 
作的泞卞母分別足 C、R、U 和 D/ 匕|’1也被你 々CRUD 操作。这叫 
都足以数倨库为中心的应 m 屮仆: 常抨遍 的操怍 烀遍到 Rails 提 
供-种方式就吋以快速生成你 ® 華的所存代码和网4^它使用支架 
(scaffolding) 来完成所 存这哼 X作。 


开始了 



代碚冰葙糍铁 

这儿有一个简单的命令，你可以从控制台执行它来生成支架 
代码。试试看你能否排列好这些代码冰箱磁铁来组成完整的 
命令。 


ruby script/generate 


ticket name: 


seat id 一 seq 


string 


seaffold . 




string ： 




|strln^J 


ad 


ddress 


email address 


" '你琬在的位釐 
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scaffold 创建代码 



ruby script/generate 


代碚冰葙糍锈斛奢 

这儿有一个简单的命令，你可以从控制台生成支架代码。试 
试看你能否排列好这些代码冰箱磁铁来组成完整的命令。_ 


scaffold I ticket ne 


seat _ id ^ se ^^| ; 


string 


3 ： 


• text 用今卷芩 
浔赛 f 


decimal 


email address 




string 


魚的啟言 


那么 scaffold 命令傲 ？ 哪些擧件噍？ 

scaffold 可以生成一些代码，这些代码允许用户在数据库中进行创 

建、读取、更新和删除数据的操作。 

.*■- - ._ j ' j ■ • •■■•...' ••. ' ■ ■** ： . * ■ ■ 

坤桌你 有 • 个以数倨 WW ^76 n - jwchi .^. rn ^ msWikm ^ 蓮行 a 建：仏 
取、史新和刪除，那么 scaffold 命令就能帑你省下很多编 V 代码的时 
M 和精力 r 

Oicket ) 削 V ;灰输入 sdold 命 让我着暑会发 

' I :什 么：- 





> ruby script/generate scaffold ticket name:3tring 
seat_id^seq : strii>g address : text pricejpaid : decimal 

一 a3dress : string 
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开始 T 




试劈 


现在该看看我 ff 】 的应用是否真的寒效了。_要肴到新的 
Ul , il : 你的浏览器衍 向：‘ ，: : 

二 ■■■«—— ■二 • •• - . -r • .... 一 ：_..... — . _ . . •• •?■•- 二 * . •• •• s —、 • 

http : // localhost : 3000 ^/ tickets 




Action Cont 


◄ [ ► C R 二 :+ ^ http:/^/lbcalhost:3000/ticlcets/ 

ActiveRecord::StatementInvalid in T 



v SQLit«3itSQLExceptioai no *uch tabl«i tickets> SELECT. * PROM "tickets - 

:.:- =■■- V /- . ••丄 〜：•： •••••• : --- - •• - ； ' - 

RAILS_ROOT : /Users/davidg/DesRtop/chap1-scaffold/tickets 




•••• •这铯咖冰 


Application Trace i Framework Trace | Fult Trace 


哪虫出错了?尽管我们 iF . 确地 生成了 
支架代玛, Web 服务器 h 还是出现 
一个错误。我们得到的是返回的一 


连串的错读消息。 


/Library/Ruby/G«ms /I .8 / gems/activ«record-2 
/Libr«zy/Ruby/Geras/1.8/g»ms/aotiv0record-2 
/Library/Ruby/Gems / 1.8/giB»8/acliv«record-2 
/Library/Ruby/G«raa/1.8/geats /act iv«r«cord-2 
/Library/Ruby/Gems/l.8/gemB/activ«record-2 

/Librairy/Ruby1.8/g*OL»/ 暴 ctiv^recorii 一 2 

/Libr«ry/Ruby/G«ms/1.8 / gems / *cliv«reeord-2 
/Library/Ruby/G«ras/1.8/geas/activ«rccord-2 
/Libr*ry/Ruby/G*a •/ 1.8/g«nta/*ct iv«r*cord-2 
/Library/Ruby/C*as/1.B/9«ns/activerecord-2 
/Library/Ruby/Geae/I.8 / gene/activerecord-2 
/L:ibr&r'y/Ruby/G«Ba/l. 8/g«ss/activ«r«cord-2 
巍 pp/controll«rs/t.ick«t*_controll*t .rbiStin 
/Libr ary/Ruby/G«ms/1 .'S/^eu/act ionp«ck 


.2/lib/activ«_r«cord/conn*ctio 
.2/lib/*oiiv«_r«cord/conn*ctio 
.2/lib/aotiv« r«cord/eonn«ctio 
i2/l ib/activ»_r«cord/conn«ctio 
•2/lib/BCtiv 看 _r_cord/corm»ctio 
.2/lib/*otiv«_r«cord/connectio 
.2/lib/aotiv« record/eonn«ctio 
.2/lib/activ* r«cord/conn«6tio 
•2/lib/activ«_r«cotd/conn*ctio 
• 含 /].ib/Bctiv«_ rttoord/base. rb ： 5 
. 2/lib/act ivc_r»coi ： (l/ba 藤 e • rb:.1 
• 2/lib/^ctiir«i_.r«coc < d/b«s«. rbi S 
ind«x a 

.•2 / 1-ib/action controllar/bas*.. rb 


/Library/duby/Gens/1.B/gens / ac tionpacV- 


^ibr*c^^by/Ce]B^dpft^ g«iu/«ctionn^k-2.1 


• 獻 ction_bontroll • 傭 • rb 

ib/action_controller/filters 



tDtrm 


考虑你在 Wd > 浏览器中看到的错误消息。你觉得这个应用为什么会出错 
呢？ 


你现在的位罝. 
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r script/generate scaffold ticket name:string 
d__seq:string address:text price_paid:decimal 
s s : s tri ngf : 


J 吏用迁移 （ migration ) 创建数据表 


数榇库中还沒布数椐表! 


这个应■应锋可弘 显杀 空的 e ； 售表 i ta # 它鮮 
什么会这样呢? ■[大 I 冷专 需要从据库中 : 个名为 tickets 的 数辋奔 
中读取这个列表， m 坫我们还没“创逮汗 m 数据 


- (4^) 


/ KkHhcaffold 命令的时候，__ 酵鄉 卿有魏顚 

构的泎细 dii r ， lAiii.fi-Ra.is'ii^r ,； 杰啾步原 则： 不要重 
复你自己 (Don’t Repeat Yourself)* iw ^ iih'.iiRails 

，佶瓜 , 你就小出柴洱 • 次提到它。 








这个数据库在 d 卜文件夹下的 development. 
sqlite3 文件里。- . 






开始了 



通过迗行迁移來创建这个数据表 


当 Rails 生成 i 架的时候，它 M 时也生成了 一小段名为为迁移 
(migration ) 的 Ruhy 脚本来创达数据表。迁移就足•个⑴来调整底 W 
数据库结构的脚本。 

观察一下 db / migrate 文件夹。你会#到 那儿冇 •个名 Mj < timestamp >_ 
create tickets . rb 的文 ft , •中的< timestamp ?^ 这个文件被 
创违时的 UTC 时间戡。如采你在文本编辑器中打幵这个文件.它？; 
上去应该就达 卜面 这个样 f •: 


㈣ 二⑽ 


列荇心.后否我们含曼 
d 女矜介绍 ii 个 


这就是为什么这 
儿 的迁杈 叫镟 " CreateTickets ^ 
它被放在一个名为 

14 ." 一 create_tickets.rb” t 

文 1 牛里。 


迁移就是一小段 Ruby 脚本。不过不要迕接运行这个脚本，你应该 
通过使用 另一个 RaihO ： 具来运行 这个脚 本，这个工 It 称为 rake •为 
rkiUa 移，在命今提示符后输入 rake db:mig 「 ate 。 这条命令会运 
行江侈 fm 汴创达数椐^ •: 


为什么迁移文件会在它的文件名里 
包含曰期和时间呢？ .... V . 











先确认你已经 mrake 命令创逑 H 尔的 tickets 数据表。然后回到 Web 
浏览器汴刷新页面： 

http ： // localhost ： 3000 /tickets 

这个 Web 应用 可以工 作了！你只焦•要儿分钟就可以输人-•些测认 
记录： 


Email address 

aian^s^angalano-COfii Show Ed‘t Destroy 
gclark<9rotlermai>.com Show Edit Destroy 
wyaU®baycity.org Siiow Edit Destroy 
eric^mananamail.ofg Show Edit Destroy 
n«lty ijpbyebyebaby.com Show Ed.t 


的几条必录, 
碎 t )5 场 0 添 


是的 我们完成了比首页多很多的内容。我们已 
经创建了整个系统。 

支架生成了整套的 W 页， It 我们可以创建、更新和 
刪除门票信息。为了弄清楚这个应用是怎样衔接的, 
让我们来创建片编辑-条新 记录。 






相 a 古 龙 你犖 up 


“ i 卜个以 (fo ⑽) .邠糾以在 

iii 


m 泰 H 的 "edit" #<2 5ibtfi-ft ai, 




/ •••••• 5这几 洩们 芍以洼棟 



你现在的位置 ► 






~ 有蠢穿 


i 河题- 

蠢碰 


有些命令以 rails 开头，有些以 ruby 开头，还 
有一些以 rake 开头，它们有什么区别_ 呢？ 

I rails 命4、是用来钊建一个新应甩的 ruby 是 
Ruby 解释器\叫来运 ff 存 妹在白 criptp 文泮夹中的工具脚 
衣 Rails 中的 / L 乎每样东西都会用列 ruby 和 rake 命令 

那么什么是 rake ? 

rake 是我们用来运行数据床迁移的命令这个 
名字的意思是. “Ruby 制造 (Ruby make )". 它被用束 
完成在其他编從语言比如 C 和 Java 中分則由 make . 和 ant 
完#的蚵类任务 ; .当 ？ rake 需要完成一个任务（比如运行 
迁移）时，它可以很机灵地分析这个应用并■决定运行 
啪一个脚本，所以它比 ruby 还要聪明那么一点点，并 JL 
它可以完成更1杂的怿务. p 如史改数掂床結构和运行 
測试 & 


我不明白“约定优 f 配罝 — 那是什么意思？ 

许多 铋枝语言提供了大量的选嘈 来供 你选择，就 
沭新车的选择方業如果'你有，个能提供大量选项的语. 
言 : 你就•需要把开戈 A 3的选择存放在 — 个地方 邋- 
常在很大的 XML 文沣里 Rails 則采用了不同的方式 t 在 
KaiIs 中 . f 物被系统焱一命名并存储在标准的位置这 
痊被称为“约定的”方式 不是因为它老鈐，乓是因 

为它4守 '約 定”或 • “标准”. 

那么我不能改变 Rails 工作的方式吗？ 

你 几乎可 以改变 R 4〖 s 中的每样穸件，但是如果你 
遵榍这些约定.你会发现开发应用的速度会快很多，同 
时其他人会觉得你写的代码史容易理解， 


，回频 rails 


复习要点 


•. 这条命令 


rails <app name>. 

为你在文件夹< app name >中创建了 •^个 NVeb 应 
用二 Rails 同时仓 II建了这个文件夹和一些文件，它 • 
•们形成 T 迨甩的基本结构-。 ，1 

Rails 附带了•一个捆绑的 Web 服务器„为了使这个 

服务器开始工作，％以使用这条 命令’ 

.. .. . '■..... . 

ruby script/server 

默认的主页位于 

http://localhost :3000/ 

Rails 应用遵循约定优于配置原则。 


为 CRUD 操作。 .. :一.… 

支 : 1 可以为你生成 CRUP 代码。要创 建用' 

: “ thing ” •嶔锯的^架 V :•.运行下面 . 

ruby script/generate scaffold 
thing 

<columr> name 1> : <column type 1> 
<column name 2> : <calumn type 2> 


_ 为.了查看你时支架，就把浏與器指向今个 URL: 

http : //localhost :3000/ things 

■ Rails 应用遵守“不要重复你自己”的原则。 

« 糝是一个脚本，它可以改变底层数据库的结 

构。通过下面的命令来运行一个 迁移： 


对数据库的创建、1取.更新和删除操作被称/ 


rake db ^migrate 
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好#!你娩救:？好朋食的工作! 


你快速完成的 Rails 应 H1 挽救 r 你的明&……至少暂时迠挽 
救了.右起宋好 像乂^ -封新的 E-mail 要处押 :了： 


太感谢你了1 

来很^ 

IS 論 

.屮 H 力有编译。 有 A 署。真的很存 


这次你真的救了我。 


: q 的标签需要让人 
, eat #” 。你能解决 


还剩一件事： 
更好辨认一当 
这个问题吗？ 


Rails 非常迅速地为我们生成 f 一个 Web 应用，这样可以为我们 
下很多时间II m 力。 m 是如采我仅I想要对生成的网页外观做 
.咋，I 、改动，阳矣怎'么做哫」 ：■ 


你现在的位置 ► 
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:支架只是为我们生成 r 代码。成代码，你就可以把这 
呰加个忡化。如恥你观察 - Kapp^'tt ^. 你么 
发观确实有很多 Li 生成的代码是你可能想要个性化-下的。 

如果你 需要对应用做一些改变，就像修改豇 iffi 的签，应 
该从哪开始呢? 


依据 Rail 的命名约定。 

还记得我们谈到 Rails 应用是怎样遵循约定 
优 F 配 置原则 的叫？它叶以让我们吏容易 
地更改应用。为什么呢？因为代码是根据 
它自身的功能来划分的。这就意味肴 Ruby 
脚本中实埤相似功能的部分会存放在相似 
的位黄。 

所以，如果你需要吏改你的 Rails 应用的行 
为，你应该可以确定需要改动的代码在哪 
儿，然后就可以更改它们了。 
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开始了 


你的应阁包宫三个鄯分： 

模型 (model). 视 © (view) » 控制器 (controller) 

Rails 应用中的绝大部分 代码棋 f F 面三类中的 -类： 


❶ 


模型代码 

模喂代码管理如何从你的数据库中读写数据。模型代码对象代表宥 
存在于系统问题领域中的东西 就像售票系统中的门票， ^ 

ii 杖1疗的在用努力去夫的 


0 


o 视图代码 

枧图是应用中展现给用户看的那一部分。因为这个原因，它有时也 
被 称力表 示层。对一个 Web 应用来说，视图代码主要生成1«]页。 


o 控制器代码 

控制器代码才是应用真正的大貼。它决定 f 用户怎样和系统交互, 
控 制苕哪 鸣数据吋以从模型中得到，以及视阁中的哪•部分用来表 
示这咚数据。 



下图显示了在一个 Rails 应用中不同类型的代码是怎样联系在一 起的: 



厂视图 

^ 枣由冏$ 诏硪. ^ 
命用 〆 f C 了在用: 
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rails 真情指数 



Rails * 真惟 拷数 

本周 访谈： 

我们询问最热门的 Web 框架，是什 
么激发他 


Head First : 你好 Rails , 很高兴你可以参加我们的访 
谈。 

Rails : 叫我 Rayn 巴。很高兴见到 大家。 

Head First : 在你如此紧张的 F 1 程屮抽出 时间定 
很难。 

Rails : 我确实比较忙。要连接数据库，要处理应用 
的逻辑，还要提供 NlH 服务，我没有多少属千自己 
的时间。但是还好啦，我有很棒的工作人员。 

Head First : 有一件苹我.直比较 疑惑： 如果你不 
介意的话，能不能告诉我们为什么在你创建一个新 
的应用的时候，会出现如此多的 n 涂呢？ 

Rails : 怎么说呢？我是一个有用的人。随笤时间的 
流逝我知道 了人 们在他们的应用屮都志要做哪些亊 
件。我+想肴到人 们遍乂 •遍地手 J : 创逮®复的 
东西。 


Rails : 拜托。我可是一个 ft 统的人。不用惊讶。一 
旦你学会了我的工作方式，你会发现我是很容钻相 
处的。 

Head First : 我听说你不喜欢波配界。 

Rails : 如果你想,你是可以配置我的，但是大多 
数人都史倾向于我咨欢的工作方式。“约定优于配 
置”原則。你明白了吗? 

Head First : 哦,是的。那正是你的设计原則中的 
一个，对吗？ 

Rails : 没错，还有“不要甫复你自己”原則。 

Head First : 还有啥? 

Rails : 不要重复你自己? 

Head First : 啥? 

Rails : 不要……嘿，你实在是太幽默广。 


Head First : 但足这样难道不会有一点……嗯 • 
难以理解吗？ 、者 J 



没存蠢河题- 

洁蠢 问骚 


我的 Web 应用中的业务逻 
辑应该放在哪呢？ 

这个嘛，得取决于你所说 
的“业务逻辑”意味着什么.一些 
人把业务逻辑定义为管理数据的 
规則 # 在那种情况下，北务逻辑 
存在于模型中，还有些人把北务逻 
辑定义为决定系统工作流秸的规則 
就诹应用包含哪些特性以及用 
户通过什么徉的顺序获取它们.在 
这种情况下，业务逻辑存在于控制 


器中。在本书后续的部分我们将使 
用“模型迻辑”和“应用逻辑”来 
区分这两神不同的情况。 

1^)* 视图和控制器之间有什么 
区别呢^ 

视图决定了应用的外現. 
而控制器决定了它是怎样工作的。 
因此，视图将会决定页面上一个 
按钮的相色以及按钮上显示什么文 
字，但是控制器会决定当桉下这个 


桉钮时会发生什么事件。 

1^)' 那么哪一部分的代码我写 
得最多呢？ 

这取决于不同的应用和不 
同的开 发者. 如果你发现你一直 
都在给应用的三部分中的同一部分 
添加代码.也许你需要仔细考虑你 
所添加的下一块新代码是关于表示 
(视图），交互（控制器）还是建 
模（馍型）的， 







开始了 


把代码描述和应用中不冋部分对应的代码连接起来。 


在线 3 -card Monty 游戏中卡片的设 I 十。 


在一个在线银行应用屮，这部 
分代码决定了你是否想把钱转 
人或转出一个账户。 

日记应用中的“约会”对象。 


在一个博客系统中，这一部分决 
定了把留言兄示为数据表还是淸 
单。 

这 部 分代码 id 录了拍卖网站的 
一次出价。 


视图 

这一部分代码决定你■要登录到 
一个电子邮件应用。 


一个链接的菜单。 


控制器 
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模型. 视图，还皋控制器 


你的 r 作足把代奶描述和 ft 用中小同部分对应的代妈连接起来。你 
做的怎么样啊*? 


在-个在线银行应用中，这 
部分代码决定 r 你是否 想把' 
钱转人或转出•个 账户。 _ 


这一部分代码 id 录丫拍卖网 
站的一次出价。 


这-郎分代码决定你畨要整 
录到电子邮件应用。 


控制器 




开始了 


这三种不同类型的代码存 
故在独交的文仵夹中 

Rails 爯欢约定优 f 配置原则并使 ffiMVC 架构。这 
会产生什么效来呢？ 

M V C 架构是怎样帑助我们去改变网飪的标签并 
调整应用的呢？ 让我们 W 看一# i 架创边的那些 
文件 • 因为代码被很淸楚地区分为三种不同的类 
型-模螌、视图和控制器, Rails 把每•类都存 
放在一个中.独的文汴夹中。 
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标签在嗖图中 


‘磨 


你的工作是标出那些为了改变页面中的标签而需要编辑的文 
件。 


©蛑戋変的差网奇的外碳.戏 r ; 軚注豸诠从 
SJL 邦些 tf 曼迕的 t 邙軚存# st 4 夹 f . 
启课名4 Mty^Ltr^O 


C3 


app 


controllers , 


helpers 

从 o 知 Lsi 4 爽中的仔何艾件， 


开‘■〕刁以泾 g i4 Hiviiws i 作 4 中的 
hfuLcrbi ^ 来更铨运安中 的杉妄. 



栽们不 f 龙诠鉍 
covJtroiUrs^ helpers S. 


视慘中的文件需要狨编辑 

如果我们想要改动网页的 标签， 我们就需要修改视图的代码。视 
图的代码总是存放住 app / views 义什央里。 

视阁文件会生成网！ a , 也被称为页面模板：那么什么是沉而模板, 
这叫模板又包含 f 呰什么内容呢？ 
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了 


编$母视怪)中的 HTML 


这4 (面揆板实际有 ' fe 來是什么样的呢？用文本编辑器打开 v i e w s / 
tickets 文汴夹中的这四个 . html.erb 文忭.文 ( t . 的内容#起来很 mHTML 。 

我们想把网签从 SeatJiUeq 改成 Seat #。 要实现这个，可以在这叫个文 ft - 
中拽索文本 "Seat id seq ". 把它改成 “Seat #” ， 然后保存你的改动。 



运样你/ 


tfb > Name :</ b> 

0 ticket . nan ^ # ^^^^ 


<lfcSeat 


luHk^^ at - id - Seq 7 <h 】 >Edi t 


<b>Address：</^ > 

<%-h @ticKet. address 


编辑完 H T M L 中的标签后，你的改 
动町以立刻在 Web 浏览器 l . iit 示 
出來。如采你观作就 完成 这作改 
动并做一次刷新，应该立刻就可 
以 n 到你的改动。让我们来看^ 
fK 面的…… 


h terror . ，f/ 

- essage s 

> .叫 

二 f f ' t e bei / ：na，ne /- 

vi 

s 撕 #” 〆 —•\ 


逐一进入到 views/ 
tichets 文件夹中的四 
个文件中，把文本 Seat 
id seq 改成 Seat # a 
这样就可以把网页标签 
改成 Seat #了。 




f * tGxt ^^idS 


； ?> < b r /> 


eg 


H 7 * 

卢没有 m 


_ 


H • 你把: seat _ id _ seq 称 
为一个符号 （ symbol ) 。什 
么是符号呢？ 


焚 • 因为它们在内存中史有效那么一点点， 

^^■号有点^•字坏巾彳符串# a 大？数情兄下，符号和字符率可以 
引号包围起来.而符号是用冒号开头互_换彳吏用 
的荇号通常被用于 Rails 中的命名. 
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现在改变，马上看见 



刷新位 f 下尚的地址的豇向: 


http :// localhost : 3000/ tickets / 


4- seot #-7. a 



/( 今狻代砝 


这是因为 Railstt 用 Ruby 做开发语言，而 Ruby 代 
叭 不需嬰编译。所以 RaiU 的 Web 服务器《 了以直 
接运行你史新沿的源代码。但是这真的很 f 不 
起吗？ 


这样 fr : 测试你改动后的代码时就减少了很多需 
嬰执 rj •的步骤了。比如你不需要编译你的代码， 
而 II 你也 H 嬰打包代码或 者在任 何地；/部署 
代码。所苻你需®做的事就是编写你的代码然 
后运行它。 Rails 的开发周期真的很迅速，而且 
对你的 Web 应用做修改也很迅速。 
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开始了 


阓0,竿上 S 点 

两次改动都搞定了，可是你的电话又响了……这次是什么 
事？ 



喂！听到应用完成得很顺利真让人高兴！嗨，听 
着……我想你应该知道我老板打电话问你住在哪儿 
来着。他很想看看你已经完成的工作，但是他还是 
有一点担心应用明天完不成，所以他想今天就检查一 
下进度。非常感谢你所完成的工作。哦，顺便问一下, 
我提到过他希望给每张售出的门票都记 录一个 联系电 
话，除了记录电子邮件地址之外吗？报歉，我忘提了 












/ \ 
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你上钩了 
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开始了 


规在达个应用霜要存储 更多的 
信息 

在你的明友提到还需要作储电话号码之前，几乎所有的事件都已 
经完成了。我 们崙要 更多的数据，这对于我们的应用来说怠味荇 
什么呢？ 



O 我们霈要在数据库中存储一个额外的字段 （ column ) 。 

我们需要在数倨库中存储一个额外的字段，但是怎么做呢？ 




乂， 


ticket . 


name 

strina 

seat id sea 

strina 

address 

text 

Drice Daid 

decimal 

email address 

strina 

Pi?one/ I 

strynq 

\ / | 

f / 


M I / I 




笔上阵 

4 . 


我们需要给数据库表添加一个字段。写出以前我们用哪种类 
型的脚本来更改数据库结构。 
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迁移改变数据库 




上一页你被要求写出我们以前用哪种类型的脚本来更改数据库 
结构。 


迁移就是 Ruby 脚本 


我们需要的是使用迁移来给数据表添加-个字段。但是迁移到底足什 
么呢？让我们回头看 T ； •那个创达 f 我们的 tickets 数据表的迁移。 



我们志要创迮的代码和这些有点相似，除了4、坫创逮数据表，而是 
添加一个字段。 







开始了 




问： 


问, 


我不1要理解这些代码吗？ 
难道对于掌握 Rails 来说，理解 Ruby 
的代码并不重要？ 

你越理酙 Ruby , 你对 Rails 的 
控制就越 K 当我们继飧洌圯本书 
的时俟，你将会学到越来越多的关于 
Ruby 语言的知识。 


没夯蠢码甄- 

/雜蠢綱 


迁移中的一些代码看起来像 
在删除数据表，为什么会这样？ 

比起我们在这儿展示的内容， 
迁移可以做更多的工作,，举个例子， 
每个迁移都有撤销它自己的能力。 
这就是为什么刨建 数据 表的代码和 
刑除数据表的代码是对应的，但是 
你还不需要对这个了解得太多，， 


问: 


如果迁移只是一段 Ruby 脚 
本，为什么我还要使用 rake 呢？为什 
么我不能直接运行这些脚本呢？ 

问得好-有痊 Ruby 是设计用 
来直接运行的，而有些不是迁移 
就不是设计来直接运行的。它们应 
该適过 rake 运行 

好吧.很好一但是为什 

么？ 

I rake 比 ruby 更聪明。当你调 
用 rake db:migratc 时.你实际上在对 
rake 说.“确保所有的迁移都被运行 
了”， 如果 rake 认为不需要，它就可 
以决定不调网这个迁移 Ruby 本身 
就不能做出这徉的决定。 


问： 


难道我不能手工编辑我的 
tickets 数据表吗? 

你可以，扭是用迁移来管理 
你的鈦据庠系仗更好„当你让应用 
上线时.你就会需要在你的产品軚 
据库中重新创建你的軚据结构。如 
策你使用了迁移.那么 rake 就能使你 
的产品数据库中的軚据结构和你的 
应用中需要的一致 e 如裟你手工修 
改軚据结构， 事 件很容 ft 就会变得 
不同步 。 就像 Rails 中的大部分情疋 ， 
如策你邊守使用 Rails 的约定，你就 
会让你自己的活儿变得更容易 


Rails 玎认生成迁移 

kM 1{, “ i 我们生成 i 架时使用 的是: 


ruby script/generate scaffold ticket name:string seat id seq:string 
address : text price _ paid:decimal email _ address : string 


generate^i -个创 ifeRuby 代糾的脚本。好消 .dgenerate 不 仅能生 
成支架代码，它还可以生成迁移。 


现在假 设你要输人这条 命今： 

ruby script/generate 


不 S 炙的铪入它 

migration PhoneNumber 


这将会生成•个全新的迁移文 fl 。 我们可以在 Ruby 代码屮加人它来修 
改数据灰。 H 题是，我们不知道怎样编写代码来完成这个迁移。 


那么我们可以做什 么呢？ 还有 Rails 可以为我们做些什么呢？ 
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命名很重要 


给你的迁移一个“聪硪”的名 
字，然后 Rails 就会为你编写代码 

你也许已经意 m 到，对于 Rails 来说命名真的很重要。当我们创建名 
为 “tickets” 的支架时， Rails 就在 http://localhost:3000/tickets 生成了 
应用，还生成了一个迁移用来创建名为 tickets 的数据表。 

在 Rails 中，命名约定很重要，因为它们会为你省略很多工作。如何 

命名迁移也同样 m 要。与其给新迁移取一个旧名宇，不如试着给它 它采用& 

一个像这样的名字： , 一衫式 4 acU *. 丁 o . 。 


11 "!' 1 ' ]，■，— —SB——i:sa————■ 

> ruby script/generate migration AddPhoneToTickets phone : string 


力什么名字会产生这些区别呢？ 

Rails 知道名叫 Add...To... 的迁移很有可能就是要添加一个特定的字 
段到一个特定的数据表中，所以与生成一个空白迁移让你填空不同 

的是， Rails 将会为你编写实际的迁移代码。 




34 第1章 










你熏要用 rake 运行你的迁移 


开始了 


这儿是 Rails 聪明地为你生成的迁移。 



rake db:migrate 


但是我们这次还能这么做呵？毕讫，我们不想 iiirake 错误地冉次运行第 
一个迁移。 


时间 戳告诉 rake 泫 运行啷 个迂移认及 1 IK 十么虓序 
运行 


Railsid 录下它 运行过 的所有迁移的 ftV ； 的时间戳。这样就 " f 以 Ihrake 知 
逬哪呰迁移已经运行过了，而哪鸣没有。这就怠味笤不论何时你运行 

rake db ： migrate. Rails 只会运行最新的迁移 c 

让我们来测试一下。内次运行 rake db : migrate 来给 tickets 数据表添加 
电话 y •码卞段。 



迗样你/ 
♦ 
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更新你的 4 图 


但是改劫数椐库还不够 

支架生成代码 这很+错， w 为它能 it 你快速就绪并运行代码。 m 坫不 

利的*而足一 K 代叽波生成，更新代码就足开 发人员的责任 广。 

我们刚刚给数据库 添加了 -个电 iS •号 w ® 性。但是因为数据衣 Ll 经被 女架 
创达 好了， 它们就不会自动地提取新的电话号码域。所以我们耑要冋到 Ui 
曲模板去添加条指向电话 y •码的引用。 
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幵始了 





格的 s — rt — 


1^* 为什么有些地方会出现 
<%=h ... %>呢？这里的 “ h ” 是 
什么意思？ 

^. # h 是一个辅助函扠。像格式 
化输出这徉的夢件就会用到 辅助凼 
It ., h 这个辅助必軚会转义域中的特 
沬字符.比如“<”和“&” • 这可以 
避免有人给网站提交包含 JavaScript 或 
者其他具有潘在危险性代码的文本 


没有 i 


?存盘网题- 

、 A 有蠢 问觀 

为什么有些地方使用的是 
字符串，而另一些地方用的是符 
号呢？ 

答： 字符串被用在页面椟板中需 
要单一文本的地方.符号（用“：” 
开头的 单词） 大多用在标签里 £ 

问：为什么？ 


符号在内存中更有效.而 
j ■大多《；接受参軚 的* 数 （比如 
f . label ) 更喜欢符号而不是字符串。 
但是在大多 ft 情况下，如果字符串 
史容易被格式化， RailsA 数元许你 
选择使用字符串代替符号 3 
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事件: 

artist • the performer (string) 
description - short bio (text) 
price Jow - cheapest tickets (decimal) 
price_high - sales price of ticket (decimal) 
event date - when it happens (date) 


你会键人什么朿 ft 数据库中创逮♦:件数据表呢? 


老板回来了 


老板对佐用的进展很满盘，同时他现 ft 沿嗜除 rid 录门 s 销忾情况，还要记录半忭 
F 面足事件的数据结构 ： 


你会 ft 控制台中输人什么命令来为事件数据创让幺架呢？ 






老板希望页面中对应 priceJow 的标签为 “Prices from” ，对应 price_high 的标签 
为 “To” ，而对应 eveldate 的标签为 “Date” 。 为了.实现这些改动你需要编辑 
四个页面梭板。把下面显示的 new.html.erb 页面模板需要的改动写 下来： 


<hl>New event</hl> 

<% form_for(@event) do If I %> 

<%= f.error_messages %> 

<p> 

<%= f.label : artist %><br /> 

<%= f.text 一 field : artist %> 

</p> 

<p> 

<%= f.label : description %><br /> 
<%= f.text_area : description %> 
</p> 

<p> 

<%= f.label : price_low %><br /> 
<%= f.text_field : pricelow %> 
</p> 

<p> 

<%«= f. label : price_high %><br /> 
<%= f.text_field : price_high %> 

</p> 

<p> 

<%= f.label : event_date %><br /> 
<%= f,date_select : event_date %> 
</p> 

<p> 

<%= f.submit ^Create" %> 

</p> 

<% end %> 

<%= link_to 'Back', eventspath %> 


在 app/v ie ws/e vents 目录中的其他三个需要修改的页面校板的名字是什么？ 


你现在的位置》 39 








又一条溟求 





老板对砹用的进展很满意，同时他现在希望除 fill 录门票销售情况，还要记录事件。 
下面是事件的数据 结构： 


:甲 rr* 

artist - the performer (string) 
description - short bio (text) 
pricejow - cheapest tickets (decimal) 

I price_high - sales price of ticket (decimal) 
I event_date - when it happens (date) 


你会在控制台中 输人什 么命令来为奉 件数据创违支 架呢？ 

?u6y sctipt/generate scattold event attist : sttin§ desctiption : text 
ptice_Loh ) : decimal ipuce_hi^h : decimal eventuate : date 


你会键入 H •么命令朿在数据库中创达囀件数据衣呢？ 


}ake d6 ： mi ^ 2 ate 









开始了 


老板希望页面中对应 pricejow 的标签为 “Prices from” ，对应 price_high 的标签 
为 “To” ，而对应 evenudate 的标签为 “Date” 。 为了实现这些改动你需要编辑 
四个页面模板。把下面显示的 new.htmUrb 页面模板需要的改动写 下来： 



在 app/views/evems 目录中的其他三个需要修改的(面模的名字足什么？ 


ddlt.Htkvtl.crb, show.htkM.l.crb^oliA^gx.htkM.l.crb 
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完整测试 


试劈 


现 ft 应用在忾票沉面 h 有所有的联系信息了: 


m http / /localhost 3000/nckets/l/edit 


所行这咚事件的信息也 被记录 F 朿-广 


Editing ticket 


U. M 


List >ng tickets 


Address 


37 NmM>ury Roac. 
Ashfield. 



开始了 


眘乐会的门累销售一空! 



整个星期应用都运行得很好,而且下周五晚 h 演#厅的 
所有座位都卖掉 r 。 


。段！门笨很怏铋卖光 
3 ……仗体们，准备好采一 
些庆议动作7码？ 


你可以使用迁移来修改你的数据表结构。使用 
如下命令来生成添加字段到数据表中的 迁移： 


复习要点 -- - —— — 

■ RaiN 遵循模型视图控制器架构，也就是 ■ 
MVC 架构。 


RaiLs 为模型、视图和控制器代码生成独立的文 
件夹。 

你对应用做的任何修改都可以在你保存修改并 
在浏览器中刷新页面后立刻呈现。.这是因为 
Rails 使用 Ruby 搭建应用.它不需要编译 . 


ruby script/generate migration 
Add<column>To<table> 

<column> : <data type> 

使用如下命令来运行 迁移： 

rake db : migrate 
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r 你的 Rails 工異箱中的工異 

5 你己经把第1章收入囊中了，现在你已经将创 
建 Rails 应用的能力加入了你的工具箱 


^ails X S 

rails app—name 

舍 J 達一个在用 

7u 6}/ sczipt/se7vet 

幵始这个 S 用 

2ub}f script/^enetate Scatto^d ... 

衿權型 ，主威 CRUD 代鹆 

7u6y script/generate mi^Tation 

t 威一个苟以设整数昶洚结构的这移 

7 ake d6 ： migrate 

4 敎拇虞中 ii 行迕移 




2 超魆支架 


+ Rails 应用，生来有€ 

+ % 



到底 Rails 干些什么呢？你已经见识丫支架是如何生成-堆堆的代码炸 
以炝鬼般的速度帮你编 KWeb ^ Ml 的，但足如果你想•史做•焱变化的话 
会怎么柞呢？ &本饫中你会孖到如何 W 正地控制你的 RaiU 开发 ，汴 11 深 
人地探究框架的吡展。你将要学习 Rails 如 H 决定运行哪呰代码， 如何从 
数据库 中淡取 数据.还心网页足如 M 生成的。 Aiin *, 你就可以 ra 你想要 
的方式來发布数椐 r 。 


这是新的一章 




MeBay 公司是一家帮助人们在线转让闲 S 物品的销售公司。他们盂 
要一个新版本的网站，而且他们需要你来帮助他们完成。 

为了在网站上发布-条广告，卖家拨打 MeBay 公司的对方付费电话, 
&知他们卖家的 ID 和想的 U : 物品的具体诏息。 McBay 拥奵自己的数 
据输入系统，你的应用需要能 ft 线发布 Mebay 的广告。 


MePay 在一个数梅库中存储广告 

所冇的广告都 ti 含同样类型的信息，而 McBay 想要在-个数据库中 
这呰广告。他们会把数据输入到你创违应用 IH ■生成的那些数据 
表屮。他们耑 耍类 似下面这样的 东西： 


的 S ! 孩 
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为这个网站使用支架会有问题吗？ 


由网页组成，可以让用 


(•■aa. < te<« 

ImQ MHt "OW • - >•««• 


数据。 


Listing ads 


笔上阵 


假设你要为网站使用 Rails 支架 a 试试在下面的架 
构示意图上填空吧。 


Nsaui »auc>«iM 

P«Un«tlMI V «-« . KtUn 卜 ‘ 

MM. J7IS r% 


首先，你要使用这条命令来创建一个新的名为 
mebay 的 Rails 应用： 


数据库模型 


包含了应用的逻辑 


















假设你要为网站使用 RaA 支架,试试在下面的架构示 
意图上填空吧 s : - 

首先.你要使用这条命令来创 建一个 新的名为 meha> 
的 Rails 应用： 


由网页组成， 可以让 




Editing act 




用户.甸.卑 


iisz 


' 

»J Sc - 

•…软据; 



为个网站使用支架会有问题吗？ 

.... 免广.黃..:....样,?17 •.费 唉.碎二个肩.辱 .的汶 蜂令* 斧作卑 .，..和, 有灼 萆: .. 
JS 触 *Msb«y t Z¥}k fMx. « fc« (S J .； 4) te<n.o.f $ £；.? r 

.... 黃 ..?..? U . .. 
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支架 


支架傲的 i 太多5 

MeBay 想要的应用比一个: fc 架式 (scaffolded) 应 Ml 做的私要 
少。支架是很帱，但是冇时应用足如此冏中.，你不如手工创让 
你的应用。 


为啥这样呢？嗯，如果你 Cl (2 编％这哔代码， 成 用就会更简单 
也更容易维护。这样做也冇 缺点： 为了手 J: 达立一个 Railsweb 
应用，你耑要深入到内部汴押解 Rails 实 k h 坫如 H 工作的。 

比我们从决定你忠要为 MeBay 创达哪作代码 幵始： 


你需要把网站上的链接和 
你应用中的代码结合起来。 


为 3 >m 过夫深 

芹，你霈要琪斛 
Railspt^X^^ 
何工怍的。 


椟犁 

0， 

你黑要为 Mebay 的广告 
创建数据库中的 一个数 
据表和模型对象„ 





你黑要用控制器代码从模 
型中获取数据并把它们传 
给视图，那儿是显示广告 
的地方》 







1 h»<» m 


不像支架式应用，你不需 
要为这个应用准备太多页 
面,， MeBay 说每个广告只需 
要单独的一页来显示。 


神么侪先溶卿些代碚呢？ 


你现在的位置， 
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先做模型 


让我们从生成 Mebay 糢型孖 
始…… 

从创迚模型代码汗始的想法不错，闪为模喂中的数据结构 
会 同时影 响控制器和视图。 

创逮模型代码和创逮支架非常 相似。 实际上，它们唯一的 
区別就是你把 " scaffold " 替换成了 “ model ” ，就像 这样： 


0 






> ruby script/generate model ad name:string description : text 
price : decimal seller id:integer email : string img url:string 


模型生成器命令将在 app 和 db 子文件夹中创 
违两个关键 脚本： 

■ 模型类 

( app / models / ad . rb ) 和 

■ 数据迁移 

( db / migrate /..._ create _ ads • rb ) 0 

迁移足•个能连接對数椐并为广告创逮 - 
个数据衣的 Ruby 脚本。为了运行这个脚本并 
创达数据表，我们志®用到 rake 。 


，: i 签扣我第1旁令皱矜辠 
4 柊 rfltec 古咚找的闳歡采 


初以含 6 iSi-J st 在用时 f » 访 


□ app □ 

I~ > 「^1 models 

U 0C 


ivc,rtatt 


V0 

... create _ ads.rb 

. ^ 

5 今名含辛尹一个 

• •时.句 if 窗续 • 
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支架 


然后我们就用 rake 真 i £ M 创 
建数椐表 0' 

为了创边数据表 • 我们需嬰 Wrake db : migrate 命令 
来调用 迁移: 



i 己住 , rake db : migrate 命令会通过你刚才在模型中创建 
的. • ._ create_ads • rb _ f 1 1 1 12'1 ii ! -1、 数則 A 。 


但是，如果你仔细査 fi •这个创逑出来的数据表,你就会发现 
件奇怪的 亨情: Rails 在数据表中还额外创逮 T * 三个字段。 




这些狄足 ： id 、 created _ atf | lupdated __ 

at。i d 字段适一个生成的主键 ， ifncrea ted_a till 
updated _ a t 则 r 数椐足何时被输人成被电新的。 


Rake 为你在数据库中创建了一个数据表，但是它并没 
有为你在表中填充实验数据。 

在我们进入下一步之前你蓠要在这个数据表中插入一 
些数据。幸运的是， McBay 公司的好心人在 Head First 
网站为你保留了一份他们的实验数据。把你的浏览器 
指向下面这个地址 

wwv ;. headfirst labs . com / books/hf rai is 

去获取一份完整的指示和数据^ 

确保你完成了这项工作，否则你后面会遇到问题 
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生成你的代码 


但是控制器是怎#的艰？ 

校型本身没什么作用。你需要•些代码来处理模型产生的 
数据，而这些就是控制器的工作。 

就像支架和模型一•样，控制器也有它己的生成器。使 
用下面的 generate controller 命今来生成一个空的控 
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超越支架 




1^) • rake db:migrate 命令总是会添加那些神奇 
字段吗？ 

没错，它一定会这么干的。 

1^)" 既使支架创建的数据表也是这样？ 

是的 a 如果你检查了前一章中数据库里的 
啟 据表.你会犮现那儿也有这些神奇字段. 

有什么方法可以让我打开数据库并检查那 
些数据表吗？ 

有的 但是你需要一个 工具。 Firefox 有 
一个叫做 SQLhe Manager 的祐件可以打开和读取 
Rails 所使用的 sqlite 3 文忤。 

• 我注意到在 generate model 命令里你 
用的是 ad, 而在 generate controller 命令中 
你用的是 ads 。 这么做是有意的吗？ 

是的。在 Rails 里.模型的名字都是单数 
的，而控制器和钕据表的名字是复数的。这意味 
着当我们用命令生成馍型时，用的是单敫名称 ad . 
而当我们用命令生成控钊器时.用的就是1数名 
称 ads 。 


蠢\碰 
问： 


这样做重要吗？ 

非常重要’ Rails 就是运行在这些约定之上 
的.所以你遵守这些约定也很必要。如果你不这 
么做. Rails 就不能正确地为你建立 Web 应用.而 
且有痊东西可能就不能工作了 „亨情会变得非常 
容易，如果你邊守 Rails 期望的那些约定的话。 


梯型的名字是 | 
¥ 系势辗 泰的劣 


我们 B 经创建 好了糢 1和控制器，现在让我们狻著采看视 ©吒 
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页面模板 (大多)是 html 


视礅1通过贞 i 糢板 ^ _ 

创建出来的~ 

T 

我们需要创建哪些视阁代码呢？ NteBay 网站只需要一个单一页面，而这一 
M 将会被用作这个网站 h 所有广告的模板。因为这个原因， Rails 中的页 
面常常被称为页 面镆板 (page template, 或者商称为模 板）。 

N 豇是由嵌入式 Ruby (Embedded Rudy， 波称为 ERb> 从模板中生成的 m 
这也是标准 Ruby 库的组成部分。如果有人需要3号广告， ERb 就会用页 
_模板以坆从模型获取的数据为这个广 为生成 HTMLMUf。 


那么 ERb 是如何产生网页的？ 


嶔入式 Ruby 含认俅令村象 
中菝奴产苦盎择 . 



吞赉徬叛嗖鳟给嵌入式 
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支架 



页 i 模板包 t 5 HTML 


当你生成了模型和控制器, Rails 就生成 f Ruby 代码。 然而视图 
釘一点不同， Q 用有一个 HTML 界面，所以视图代码是用 HTML 
写成的也坫行逬 押的。 


为了创建这个广 A •模板，打开文本编辑器 II ••创让-个名为 show , 
html.erb 的文件，然后把它保存在 app / views / ads 路 控下. 
你需要 itshow.html . erb 的内容 I ；起来像这作： 


创建 sh 6 w • html : erb 女件并 
把这些代码添加到这个文件中。 


银 Z " 9 矣 Ruby 

(ERb)J/-m^ 

创建 W 沁页 


此时此刻这 t 模板眘起 来还足 -片空丹，但足不久后你就会召到 
抢制器 迠怎样聪明地在它 liWWi 人数值的。 

那么，实际的 Web 应用看起来是什么样子的？ 




路由是必需的 

劈 — 

用这个命令来启动你的服务器 

ruby script/server 

然后把浏览器指向 


http://localhost:3000/ads/3 

发生了什么? 



Web 应用出错了。怎么 回參？ 

我们刚刚 f - 工创建了 •个 Web 应用的基本骨架，但是我们还没軒 
告诉 Rails 怎样使用这个新的 show.html .erb 模板。 
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我们应该怎么做呢？ 





超越支架 


路由 # % 诉 Rails 你的 
在哪儿 

Rails 需要•条规则来指明对干指定的网址要运行哪些代码。这是 
Rails 中为数不多的几件真正需要配腎的赛情之一。 


丄 

. 




Rails 用来映射 URL 路泾到代码的规则被称为路由 ( route) 。 Ruby 
程广 f •中的路由是在 con f ig/r ou t e s. rb 中定义的，我 ff j 还需要为 
show.html . erb 模板添加一条新的 路由： 




如果布人请求 / ads /3, 
我钬用 ads 控制 S 和 show 
找板，再拕参数 id 设为5。 


config 


routes.rb 




ads _ controller.rb 


这儿黑体标出的路由会匹配 


http: //mebay. com/ads/3 


用 ads_controller • rb 作为控制器代码, 
页面模板。 


但这些到底是怎样工作的？发生了些什么呢？ 


show . html.erb 
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路由负责 URL 匹配 

幕后英雄是路由 


0当 Rails 收到一条来自浏览器的请求后.它会把请求 
路径传递给 routes . rb 程序来找到一条匹配的路由。 


map.connect '/ads/ : id' , 

: controllers'ads', : action=>'show 




' ••… ^ ^ Mttp；// 

^taLj.&ov^/aoig/sttl ^ ifi 

I GET/ads/3 

0 

routes.rb 



如果匹配的路由包含符号.那么路由系统就会在 
请求参数数据表 params […]中创建匹配的参数- 
这儿的符号.我们指的是由冒号开头的一个字母 
序列。 


子邊 4 糾罨个浔芩 

map.connect '/ads/ : id' , 

: controller=>'ads', : action=>'show' 



Q 路由也可以指定另外要插入到 params […]中的 
参数。 

: controller 和: action 常常在这儿被指定。你能想 
到为什么吗？ 


map.connect •/ads/ : id ’， 

: controller=>'ads', : action=>'show' 





/ 



| GET/ads/3 

0~ " 

routes.rb 





Name 

Value 

: id 

3 

: controller 

'ads' 

: action 

show 
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支架 


Q —旦 routes . rb 程序结束， Rails 就会查 

看 params [: controller 】 的值，并用它来决定自己需要 
创建什么类型的控制器对象。 


Name 

Value 

: id 

3 

: controller 

•ads' 

: action 

show 



O —旦控制器对象被创建了， Rails 就会使用存储 

在 params [: action 】 中的值，来选择要调用的控制器中 
的方法。 




def show 
end 


ads_controller.rb 


Q 当控制器方法完成后， Rails 就调用同样匹配 

params [: action ] 值的页面模板这个页面模板会生成 
要返回给浏览器的响应。 



Name 

Value 

: id 

3 

: controller 

*ads ' 

: action 

show 
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深入思考 


卞没苕蠢離 

A 苕蠢舰 


生成支架然后再修改不是会 
更快吗？ 

这会因不同的应用而不 
同。 MeBay 只想要非常少量的功能。 
如果你的应用需要实现和 支架栽 然不 
同的工作，只生成蟆型和控釗器.再 
添加你自己的代码会快得多、 

问： 支架也会生成模型吗？ 

是的支架生成器会调用馍 
型和控制器的生成器。它也会为标准 
的刨建、读取，更新和刪除操作钊建 
页面馍板。 

1^)- id 是什么类型的参数 9 

答： 它是一个请求参 ft . 就诹被 
表单提交的值，或是那些传递给 URL 
的参敫。 

什么是请求？ 

请求就是备你点击一个链接 
时浏見器发送给服务器的内容.它会 
告知服务器汴需要的确坊路後。 

那什么是响应呢？ 

哙应是服务器返回给測莧器 
的内容以及其他的一些信息，比如内 
容的 mime-type t . 


为什么 Rails 黑要路 由配罝 
呢？为什么不是直接使用标准路径 
呢？ 

比起配置. R a i 1 s 总倾 
向于使用约定，除了系统需要 
和外界对话的时候 。 URL 的格 
式影响卟部世界看到的应用的 
徉子，所以 Rails 让你配置它们。 
当你使用支架的时候， Rails 为你生成 
路由，但是理铒路由是如何工作的纣 
追踪出锚或创建自定义路由还是很有 
帮助的，比如在这个应用中。 

1^)* 我还是不太明白何时使用骆 
驼拼写法。到底是在什么情况下使用 

呢.•， 

骆驼耕写法 （ CamelCase > 的 
意思就是在包含了多个单词的标识 
符中使用大写字母。它有这个名字是 
因为那些大写的字母看上去倬骆驼的 
驼峰。 

在 Rails 中，文件名和控制器名看上去 
很相似，但是控制器名使用骆驼拼写 
法来分际单词，而文件名使用下划线 
来帮助你区分代码和文件。 


先调用的是 哪个： 控制器还 


是视图 '? 


控制器总是在梘图之前被调 


^1- 为什么页面横板的后缀名是 
.html.erb? 它难道不只是一个 HTML 
文件吗？ 

一个橫板可以是一个简单的 
HTML 文件，但是很多糢板会包含 
额外的指令，这痊指令会被嵌入式 
Ruby 系统处理所有你想要嵌入式 
Ruby 处理的那些文件的文件名末毛 
都有 “. erb ” • 

i 1 ®)' 为什么模板在一个名 
为 “views/ads” 的文件夹里，而控 
制器却不在 “controllers/ads” 文件 
夹里？ 

想象一下你要编辑并查 
看一个对象 • 应该会有一个“編辑 
( edit ) 页面和一个“查看 ( view ) 
页面 u {£ 是 “ edit ” 和 “ view ” 两个请 
求都会通过同一个控制器。所以多个 
馍型会有同一个控制器，但是可能会 
有多个页面。这就是为 什么页 面樓板 
在它们自己的子文件夹里，可能会有 
多个页面模权„ 

I ®).*我听过人们谈论■•业务对象 
(business object)” 和“领域对象 
(domain object}” 。 Rails 有这些概 
念吗？ 

有的，因为业务对象和领域 
对象託是模型对象的别称。 




支架 




♦ 命 j 


MeBay 的竞 争对手 巳经有一个 Rails 应用 r, ⑴的就适下向这组路 Hh 

map.connect */shows/ : title 1 , : controller => * shows *, : action=> ’display’ 

map.connect '/cats/ : name', : controller *> ’cats ’， : action=> 'show* 

map.connect '/gadgets/ : type *, : controller => 'gadgets *, : action=> 'show / 


你能找出哪个页面校板文汁•被川来冶指定的 URL 生成 HTML 叫？ 
画线把 URL 和会被用到的页面皎板迮抟起来，然后 写下从 甸个 
URL 中提取的参数的名卞和值。 


从莕 个两份到吋在的毋 

http://yourbay.com/gadgets/display 

参 数名： . 值： . 

http://yourbay.com/shows/cats 

参 数名： . 值： . 

芗下 UL 备个 认亂 中砂的 
年致的名多扣 $ 

http://yourbay.com/cats/gadget 

参数名： . 值： .. 




□ 


^0 

display.html.erb 




3 

_ 

^9 

display.html 

ShOM 

□ 

3 

Iay.html 

U !^ 


show.html. erb 


gadgets 


show.html .erb 


shows 


display.html. erb 


I ^ } 

show.html.erb 
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我的路由是什么？ 


我的牌由龙1+么？ 

MeBay 的竞争对 f - 已经有-个 Rails 应 用了， 用的就是下面这组 路由: 


map.connect ' / shows/ : title' , : controller => 'shows', : action=> 1 display 
inap.connect '/cats/:name ', : controller => ’cats ’， : action=> 'show 1 










支架 



试劈 


用浏览器打幵这两个豇南 i: 

http://localhost:3000/ads/3 以及 
http://localhost: 3000/ads/5 


资 $ •:泛 笱铉兹 


， ► c 

Name: 

Description: 

Price: 

S«Uer Id: 
Ematt: 


http //locaiho&t 3000/ads/S 

♦ 4 hno ; . lotalhost 3000 /ads /5 




Htto //localbost 3000 ； ads/3 

C < :i + ^"np r^tocaihost 3000 /ads /3 


Name: 

Descriplkin: 

Prk«: 

Seller Id: 
Email: 


㈣ .:多 


Q - 


广皆丧空令的, 


— 姑 ―:-;- 

为什么页面中没有具体内容？路由有问题？还是模型 ？- 
视图？控制雜？ _ 
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视图 + .. ._ 数据 


视餍沒有要显示的数椐 



回颐一下 show.html . erb , 这个文件被用来 
为每个广告创让豇面 而这 价好就边模板 Li 
经完成的 工作: 


app 


1 ads 

l - ) 


show . html.erb 


尽苻我 ff ] 已经放好了 HTML 的主要 竹架 fe 签、 
主体和标题部分，但还是冇好多售情被湘掉 r 。 
我们还没有 指明： 

志耍 趾示哪 些数据 

或者 

数椐 ® 耍被插人到页 iTd 屮的哪个地方 
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超越支架 


那么页 i / S 泫温示哪些沟容？ 

我们：要广告 M 面 K . 承 URL 中桁定广舟号的数据。例如，这是3号广告对 
应:的 URL : 


http: //localhost : 3000/ads 汽 、每 望 d 个 hrlS •子 3 多广苦 

含了 . . 广苦致邛表.齐押 

fe 倉） 、 . 〔以 = 3 的记录 — 


mm 


description 





OK : 


Old manual typ... 

bsb 

54 

dhamjnett@email... 



；■ • i … 


mm 





Slightly 

moth 

1 -. . . 2978.25 56 kathy@hotm 

ail.... http: 


//saloon.... 


你需要做的第 一 fl •亊就足告诉校型从数据库屮 
的 ads 数据衣中读取 i 己录， id 录号和 URL 中的 
id 号相同。如果用户请求的页面是对应3号广 
告的，校型就志要被告知去读取 id = 3的 i 己 
录。 


我们需要在正碥的忮 g 
S 示数鴂 

读取数据只完成 f 一半。一 r 模型读取 r 数 
据，它就需要把数据传递给视图。接苕视图 
■要知道在每个页面中的哪些位置 M 示这些 
数据。记录中的每个域都需要被迠示在网页 
中对应 的标签 旁近。除此之外，你还需要使 
用 img _ url 字段中的值来在页面中插人一■幅 
待 tt •商品的阉片。 


那么，系统的哪一部分负责从数据库中请求 
相应的数据并把这些数据传递给视图呢？ 


_ 

I < » http7/localhost ： 3000/ads. 




ooschead 


Ascription tl 

Ah 


lightly moth - eaten . One of the antlers is broken and 
ere's a strange buzzing sound behind the eyes ... 


12978.25 




athy @ hotmail.com ^ 
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控制器与视图合作 


控制器拕广告岌送给视楹 


让我们来看看控制器中的那些代码的样子以及它们是怎样工 作的: 


0当用户的浏览器发送对某个页面的请求给应 
用时， Rails 调用 routes . rb 程序来确定需要运 
行哪段代码。 



\ 

•rb 


routes, rb 



O routes . rb 检查请求的路径并确定应用需 
要使用广告控制器来执行一个“显示 
( show ) ”动作。它也会使用请求路径中 
的值 “3” 来创建一个: id 参数。 



rouctts.A 也食备) 

達-个扣的 flds 控判器 





M 崎 3 


params [ : id] 


0控制器 看到: id 参数被设置成 “3”，所 
以它要求 Ad 模型找出 id = 3的广告对象。 
控制器告诉模型使用称为 “ finder ” 的 
方法。 



0 

routes.rb 







Ad.find(params[ : id]) 
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超越支架 


A Ad 模型从 ads 数据表中读取 id = 3的记录并把 

结果返回给控制器 ' 





Ad.find(params [: id]) 


O 控制器通过把 3 号广告的数据赋予名为“ @ 
ad ” 的变星来把数据存储到内存中。页面模 
板可以访问 @ ad 变量.这样它在生成网页的 
时候就能使用广告数据。 


if =/ 


i 

@ad 


0 


Ad. find (params [ : id]) 

, i 4 -个紗 8 H 






ads _ controller.rb 中已 A 成的代码需要被 

放到一个名为 show 的办法中去，这个名字用 
来 Hfldroute . rb 创速的： action 参数的名卞。 

但是模型到底是怎样从数据库中读取数据的. 
以及页面模板又将怎样使用这些数据？ 


达样你 / 




r 


ads_controller.rb 






把上面完整的控制器 
代码输入进去 。 w 


% % "sViaw" 


你现在的位置 ► 


67 












从记录到对象 


Rails ■杷 ia 彔转化成对象 


3Rails 从数据库中该取 nURL 屮 id 的 iti 录时， Id** 中的数据就敁转换到 
一个对象中。这个对象被作 figfr: 内存里，而II控制器会给它取名为 @ad。 



但是记 录有好儿个域，毎个域屮都符数据。所符这呰数椐蛙如何存 WTfr: 
同一个对象中的呢？ 


答案是-个对象可以冇奸几个属性。铋性就像记录中的字段。它冇名字 
和值。所以当 Rails 从数据库的记录中谈取到 description 的值时，会 
把它存 fiSifthd 对象的 @ad.descriptionW 性中。冏样的事发生在 id、 
name, seller-id 等戈似 的氷悄 h。 

通过这种方式， @ad 模型对染完全匹配广数 



雜 EESHW ES^STa^Simillil EL 

3 Moosehead Slightly moth-... 2978.25 56 kathy@hotmai1.... http://saloon.... 
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超越支架 


数椐在沟 存中， 而 R 页玎吆看 见它们 

页面梭板 ( show.html.erb ) 并不 A； 接返回给浏览器。首先它会被嵌 
入式 Ruby 程序 ERb 处理.而这就是为什么我们的模 K 有一个 .erb 义件后 
缀名。让我们仔细#行 ERb 足怎样从内存中读取对象的。 

ERb 遍 W 整个模板来 A 找被称为表达式的一小段嵌入式 Ruby 代码。毎个 
表达 A 使屮<%=和％>«起来. ERb 将把表达式替换成它的值。所以9 
它在 M 页的某处 找到： 

<%= 1 + 1 %> 

时， Kails 会在把页面返 M 给浏览器之前⑴2来婷换这个表达式。 

仉 是我 们真正要实现的足从内存中茯取 @ad 对象甩的情，就像 这样： 




<html> 

<head>^ 

<title><*= @ad.name %></title> 
</head> 

<body> 




<b>Name:</b>< = @ad.name %> 

</p> 

<p> 

<b>Description:</b><^= @ad.description %> 


<P> 


<b>Price:</b><%= price %> 

</p> 

<p> 

<b>Seller Id:</b><%= @ad.seller_id %> 
</p> 

<p> 

<b>Email:</b><%= @ad.email l> 

</p> 

<P> 

<img src* M <%= @ad.img_url >>"/> 

</p> 

</body> 

</html> 

在_ 轉责面之前， Rails 会把所冇的 < =... 标签替 
换成它们的封象值。 



那么 —— 这样有效吗? 
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试驾 



为 f 测试这个系统， MeBay 的「.作人员爪他们的数据输人系统来向 
Rails 的数据库插人数据。 

数椐一旦存储在数据库 中就可 以在 Web 上看到了。所以如*:有人 W 
求 / ads /1、/ ads /2 等，他们就会看到由数椐库中相应数据生成的 
网页 • 




你已经完成了第•个 T ：LRaiU 应用！尽管它 ff hi 比使⑴支架生 
成的长一点，但迠每一步都在你的控制之中。史®要的追，你初 
步接触了 RaUs 的内部机制，并学到了它的•部分 本领: 
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支架 


代碚泜键越 



ay 希望用 / sellers /: id 显示卖家的信息。用 
代码池中提供的代码片段完成控制器和 
页面模板。 


def stats 


end 


« Seiler.find( 


<p> 

<b>Number of Sales:</b> . 
</p> 

<p> 

<b>Total sales values ： </b> 

</p> 

<p> 

<b>Average price ： </b> 

</p> 


提示： 代码池里的每 
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完成代码池 


代碚泜键拯解著 



MeBay 希望用 / sellers / : id 显示卖家的信息。用 
^代码池中提供的代码片段完成控制器和 
ir 页面模板。 




.一 


^ __ u_ _ 

<b>Number of Sales ： </b> @seller.num__ } CD- 


<b>Total sales values ： </b> 

</p> 


Q 


@seller.total 一 sales 


4 CD. 


f 心 U 


<b>Average price : </b> 


total 一 sale JL^JL . ^]CZ} 
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支架 



为广帮助用户看到郎冇哪咚广 ft , 沣 H . 
帮助他们从中找到他们感兴趣的那鹓广 
告, MeBay 的工作人员要求有一个索引 
( index ) 页面来 InL 示到所冇豇 向的 链接。 


假设你想创建一个新的名为 “index” 的页面，如果你想用 http:// 
mebay.com/ads/ 来调用它的话 . 路由应该是什么呢？ 

控制器中被调用的代码是什么？ 

页面模板呢？ 


你现在的位置， 
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创建索引页面 




没存舰甄 

甫蠢问题; 


什么是动作 （ action ) ? 

动作就是 Rails 应用发起响应用户请求 的一系 
列 操作， 动作参敫指定了动作的名称。你的所有代 
码（比如控制器中的方法 以及页 面樣板 文忤） 都使用 
了动作名称，这徉好让 Rails 找到它们^ 

我能在 Rails 中使用任何数据库吗” 

所有的主要数据库.比如 SQLhe 3、 MySQL 和 
Oracle . 都是支持的.另卟.大部分情况下你不需要 
写很多致据车特定的代码这就是说.你可以在榖据 
卑系统之间 b 换，而不会破坏你的应用或需要重写大 
量代码 


像 Java 之类的编程语言除了对象还有原语 
( primitive ) 0 Ruby 或者 Rails 有原语吗? 

没有. Ruby 中没有垛语在 Ruby 语言中你处 
理的所有事情（包括啟字和代 码块) 都是对象 

页面模板是不是就是网页的另一个好听点的 

名字？ 

不是页面楝板是用来生成页面的.它并不 


不是 
是页 面衣身 


页面馍板是用来生成页面的.它并不 
冋页是用页面撗彳反生成的。 
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超越支架 


笔 


上阵 


{爯一次) 


有两条 路由： 

map.connect 1 /ads/ : id', : controller ^ 1 ads’ ， : action=> ' show 
map.connect */ads/' r : controller=> * ads', : action=>'index' 


每个 URL 会显示哪个页面呢？ 


/ads / 多 



/ads/somethmg 

/ads/ 

是不是哪儿有问题？如果是，该怎么修改呢? 



l)rwri|>lion: SIi(|mI> mnlh-«iilci> Onr ol thr aiMk-f' hnAcn «ml 
there 、 * bu/rmf xmiimI Miinci the eyc\... 





show.html.erb 
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迷惑人的路由 




H 再一釗 


有两条 路由： 

map.connect '/ads/ : id*, : controller=> * ads', : action=> * show' 

map. connect •ads", : controllers * ads *, : action->' index * 

每个 URL 会显示哪个页面呢？ 



Vflds /" 路踔倉罔的 西紀* 条硌由. 
f 碎诠威一条珞 t 




路由按1优先级顺序运行 

两条路由邯卩 Lj^/ads 路砭。 Rails F 避免校•棱两 W 的怡形 Rtt ⑴第•条匹配的路由。 
以需要敷新徘列这吵路由来解决混淆的 H 题。 

map.connect '/ads / 1 , : controller^〉•ads * , : action => 1 index' 

map.connect •/ads/:id 、 ： controlier=>•ads•, :action=>'show* 

有一些 Rails 将耍使用的路山，观 ff / 若要你来完成这呰代码。 





把这些路由加入 
config / routes . rb 。 
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超越支架 



踣由真傦抟努 

本周 访谈： 

作为 Rails 的主要交通枢纽，生活是怎么样的呢？ 


Head First ： 你奸.路由先生。很荣幸你可以为我们腾出 
一些你的宝贵时间. 

路由：不， ads 控制器 . ads . 没错，就是 那个。 

Head First ： 路由 先生？ 

路 由：喂 tt 边，老圮。淸求躭要来了……1嘟嘟……嘟 

嘟1 

Head First ： 石来你的 r 作真的很忙。问题适，尽答•你在 
Rails 应用中占椐荇作常重要的地位，有呰人还足搞不清你 
到底做些什么。 

路 由：哎 ——我不蛏为 f 知名度来做这份工作的。引导和 
服务，那就垃 我。 我鱿像一个交通焚察，明 A 吗？ 一个请 
求从那边的那个门过来广？ 

Head First ： 什么——端 n (port) ? 

路由：是呀. 这个眼务器上的是什么？端 U 3000 在那儿。 
这个请求是从 Webiilj 览器过来的， H 标一一我不知道一 
大槪足 / donuts / cream 。 

Head First ： 然后呢？ 

路由： 但是 Rails 不知道该运行哪段代码来给它提供响应。 
所以他来找我，而我会査看 / donuts / cream 并把它与我 
在这儿得到的路由清单对照…… 

Head First : 哇，有好几个呢 • 

路由： 是的。所以我从头开始检资这个路由清单并寻找 
第一个看起来和 / donuts / cream 相似的路由。我可能 
找到了 . 比如说 / donuts / : flavor . 

Head First ： 这个路由确实很相似，但这怎样帑助你将请 
求引导至正确的代码呢 J 

路由 ：嗯， 毎个请求都会带音文件来找我填写》—系列调 
用请求参数的名称和值， 看 到没？ 

Head First : 哦，是的 • 好多 条目. 

路由 t 没错.所有的请求都有这些。它们被称为 


params [...] c 我査 fi •这个路由，然后它会告诉我匹 
fti / donuts /: flavor 的毎条路抡都需要使用 donuts 
校制器，比如说， display 动作。 

Head First ： 这么说梃淸楚的。 

路由： 所以我在 params 【 • • •] 中添加史多的内容， 
比如值为 donuts 的 params [ : controller ] 和值为 
display 的 params [: action 】. 

Head First ： ……然后 Rails 使用这些来选择要运行的代码。 

路由： 完全 正确！ 你卞得很快.伙 il ! Raiha 兑： “哦我明 
白了. 我黹要使用制器。确实我应该创逮一个 . ” 

Head First ： 确实? 

路由： 也 I 午不是确实。但是不管他说的是什么，他 
知道他需要创迮-个 donuts 控制器对象。而且因为 
params [ : action ] 被设置为 display , —旦 donuts 控 
制器对象生成了，他知道在它上面调用 display 方法。 

Head First ： 那么你提到的路由中的： flavor 会怎么样 
呢？ 

路由 ：哦， 那个嘛，是的。如果这个请求要的 
是 / donuts / cream 而这个匹配 / donuts / : flavor , 
我就会添加另一个参数，也就是 param [: flavor ] = 
1 cream * 0 所以我就会记录下要求的是什么，如果它 
对于后面的控制器中的代码很重要的话。 

Head First ： 谢谢您，路由先生，这次访谈真的很…… 

路由 ：喂， 等一会儿！抱歉 • 这足一个神经质的家伙，他 
老&双击他的超 链接。 每次一个！毎次一个！ 

Head First ： 谢谢••… 

路由： 別 客气。 听着，现在这儿有点忙。为什么你不继续 
往前看#这个应用的其他地方都发生了些什么呢。是的， 
就沿着左边往前走……我觉得会有一咚新代码加入到这个 
ads 控制器中…… 
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控制器控制着系统 


为 J 拕数椐放入视餍，你还 f 要 
控制器中的代码 


模型已经好了，有一条路由对应干你所需要的新柠制器代 
码。还需要什 么艽他 的吗？ 

嗯，是的，你还需要做两件事。索引页面需要控制器中的争 
独代码，因为它要处理很多广告，而且你还需要一个新的豇 
面模板宋在视阁中敁示这些内容。 





lew Oot 


o< the n bnAr* «k! 
yami fcrtand tfK <) A- 



^ 1 否* ? 律托 



还需要什么其他的？控制器需要做什么呢？ 



有什么是控制器要为一个索引页面做的，而不需要为一个广 
告页面做呢？ 
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超越支架 


索引页 if 要來 f 所有纪 
录的数提 _ 

广告页面只需要来自单一记录的数据，但是索引页面怎么办 
呢？它需要读取整个广告数据表中每一条记录的数据。为什 
么会这样呢？： 

来看 一孬索 引的设 it 。 它需要为所有的广告页面创逮链接， 



你现在的位踅 ► 
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find (: al !) 


Ad.fiwd (: all) 一次读取整 
个数椐表 

Ad . find (. ..)finder 方法还有另一个版本，它会返冋整个广告数据表 
中的每•条 i 己录的 数据： 


广 ^def index 

你 *: t«ii 个汸法 @ads = Ad. find (: all) 




end 








达样炒/ 


把这个方法加入控 
制器。 


m 这是怎样实现的？毕寇，3你只是读取申-一记录的时候， ms - 
很简单的。你传递给模型 -个 id 数攸，而模喂返 M 单个 对象，其中 
包含 Hd 对应行中的所有数据。 

但现庄你不知逬你总共盂要读取多少条记录。这是不是息味右，你 
志要的代码会非常非常 k 杂？ 

幸运的蛙，事情不是这样的。 Rails 使得读取-张数据表中的所 Aid 
录和读取单一对象非常相似。当你调 mAcLfind (: all ) ,模吧返 
N 单一对象，其中包含了数据表 中所打 记录申.的数据。控制器可以 
把这个对象賦值给单一变 a 。 

m 是 Rails 怎样在•个对象中存储不定行数对应的所釘数据呢？ 


它是通过一种特殊的对象实现的。 
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数椐作为一个称为 gg 的 
对象狨返回 " 

与負:接返回一个包含申 itl 录的数据的对象不同的是， find 方法创 
建了很多对象——每条记录创建一个，然后把它们打包到-个称 
为数组的对象中。 

finder 方法 Ad. find (: all) 返问单一数组对象，其屮按顺序^含 
了所有对应数据表中各行的模型对象。 

控制器可以把单.-数组对象存储在内存屮，名为 ® ads 。 这样做对 
页面:模板來说电方便，因为不需要在内存中#找-个不知名的模型 
对象，模板 H 要知进数组的名祢，就"了以访问所有的模喂对象了。 



i 是一旦对象存储在数组中之后，你该怎样访问它们呢？ 
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ahecC7-3tiif terf> 




















数组就像很 多小桶 

数组就是一些编吾后 
的对象序列 

这个 @ads 数组在一个编过号的桷序列 甩存 储 r 梭％对象，从 
mo 丌始。毎个 fflip . 保存的对象被你为数 m 的元素。 



@ads[0] @ads[l 】 @ads[2] @ads[3 】 @ads[4] @ads[5 】 @ads[6] 


你可以用保存苕元素的槽的编号读取数 m 中毎个单独的 
元素。 


@adst41 


^夺致讼妗缚4中戊嘩的 的象 I 
10( = 5 的6栌數旖豪中的 H 


楢永远都是从0开始递增编号的， 而数组 》 f 以和需要的一 
样大，所以数据表屮到底有多少条记录并不 t 要，它们都 



数组索引从0开始 

这意味着每个元素 
所在的位置就是它 
的索引值加1。所 


W@ads [0] 包含着第一个元素， 
@ ads [ l ] 包含着第二个元素，以 
此类推。 


" r 以存储在单 - 数组对象电。 


问： 


为什么数组是从 0 开始而不是 
从 1 开始的？ 


问: 


，.- » 

丫没痒蠢通- 

々 存蠢碰 


当你在数组中放入些什么， 
数组会保持一份单独的拷贝吗？ 


问: 

呢？ 


数组实际上是不是一个对象 


这是历史原因 u 大多钕编裎 
诰言都有数组，而且大多麩情况下数 
纽的索.引都从0开始。 


答： 不会。数组只会保持对存储 
在内存中的对象的引用。它不会保留 
莱个对象本身的拷贝，而只是记住这 
个对象存在哪比。 


是的。彀组是一个完全的 
Ruby 对象 a 

一个数组能有多大？ 

对軚组的大小没有限制，它 
在内存中占了多大的地方就是多大。 
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使用数组索引 



把这些对象插入到页面中，就像数据库中只有这三行 一样: 


施 


views 

<latkvd> . 

<一 rp 鉍 “ 这个 , ads 

<tltit>All Aois<AltU> 爹合昶扣咬 4 ‘，肩一茗仿时尤 1 

<M> 科 7.‘ d “-㈣ .’光 

<body > index.html.erb 

<hi>Alt Ads</hi> 

<uL> 

<ll><a href = Vflds/<^ = @ad^lol.ld 多 >">< 多 = @flo(s[o].kv«M«cc ^></ax/li> 
<Uxa href=Vad&/<f= 多 >->< 多 = (®«ds[i].i^Kwte </a> </U> 

<Lixa hrcf=~/flds/<^= @«ds[a]_Ld 多 >-> < 多 = @ac(s[^L^^ fo></ax/ll> 
</ui> 

</bodij> 

</htWA.l> 


上面的代码 U 会显示 3 条广告。如果有4条，5条其至3 ()00 条广 
告该怎么办？你绝对不想毎次 fr : 数据库中增加或刪除一条广 
告的时候都要修改豇面校板。 



你需要某种写代码的方式来适应数据库中任意数目的广告。 
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循环代码在循坏 


用 for 循环读取所冇的 
广告 " ~一 

一个 Ruby 的 for 循环可以让你一次又一次地运行同一段 Ruby 代 
码。它可以用来读取数组中的多个元素，毎次读取一个元尜, 
并对每个元素运行一段代码。 

每次运行的同一段代码被称为 循环体 (loop body) 0 循环体 
会按顺序为数组中的每个元素运行一次，从 0 号元素开始： 


4神钚中《个无#攻 
命名冷 X 。 




for ad in @ads 


# Do something with the 'acT object 


end 






在上尚的代码中，毎次循环体运行时，数组中的当前 元素被 
取名为 ad。 所以 ad 指代的是每个 Ad 揆嗤对象，而在循环体中 
你可以访问所有的模贺对象域 性： 广告的具体信息，比如名 
称或垃商品描述等。 



现在，我们需要生成 HTML, 它会创边一个通往广告网豇的 
链接。但足 HTML 足由页面模板生成的，我们怎样用 for 循环 
来生成它呢？ 
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数组中的每个元素都 

f 要 HTML 


h 


我们可以通过一个 for 循环来实现它。循环可以 U : 我们一次处理一个广 
告。如果我们使用循环体來生成 HTML , 我们就可以为每个广告创逮链 
接： 


问题足，我们是通过把 Ruby 表达式插人页而模板来生成网页的。沉面 
揆板中的 HTML 控制荇 M 时调用 Ruby 表达式。但是我们现江想用 相反的 
方法来做事情。我们想用•个 Ruby 的 for 循环来 控制何时生成 HTML 。 
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模板和脚本段 < scriptlet) 

Rails 粑页 f 模板转换成 
Ruby 代码 

前面当我们想 ft 页面中得到对象的值，我们就会用 
<%=. • •%>来插人 它们： 


<%=@ad.name %> 


ERb (嵌人式 Ruby) 通过把校板中的毎个表达式#换成 
它们的值来从页面校板生成 N 页。 ERb 蛙通过把整个豇面 
的换成 Ruby 代码來实现这项工作的。 

想象一个页面模板中只有以下内容： 

<titleX%= @ ad.name %X/title> 

板枝•苍 C 

ERb 生成 Ruby 代码来打印出每个表达式和每段 HTML。 所 
以上面的模板代码会转换成下面这样的 东两： 



cv-b 


m 


print ，， <title> n 
print @ad.name 
print n </title> n 


o 


f " 

巧 1 U 的 


接 苕这些 Ruby 代码就会被执行，而执行结果就是经阿络发 
送给浏览器的 内容： 

由 Ruby 代硌 知工后給出 
的 HTMU 

<title>Moosehead</title> 

如果你想用同一个模板来为数组中的每个对象生成代码，你 
会让 Ruby 代码看上去是什么样的呢？ 
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玎认用 脚本段 拕循环加 
入到页¥ 1 ¥中 

此时此刻让我们先忘掉页面模板。如果你正在写一段为数组中的 
每个元素打印 HTML 的代码，那这段代码应该是什么样的呢？它看 
上去也许有点像 F 面 这样： 


for ad ift @ads 


•<li><a href= 


print •</a></li> , 


物 


〆 亇象么式 


我们需要用循环遍 W 数组，并为每个元紊打印出 HTML 和表达式。目 
前我们只看到过 ERb 生成打印命令，但是 for 循环泮不是打印命令。那 
么我们怎样才能把成段的 Ruby 代码传给 ERb — 就像 for 循环一样呢？ 

解 决方法是使用 脚本段 ( scriptlet ) 。 

脚本段就是一个包含一段 Ruby 代码的签。表达式标签被 <*=• . .%> 
包围猗，而脚本段是被 <*...*> 包围若。脚本段开头没冇=符号。 

为了理解脚本段是如何工作的，让我们用一个页面模板来产生上面的 
for 循环代码。 


埤本歿约幵诒:眘“ = 


色倉 - =- 


的表2式 # 
/ 


对象的皇用 

<%— 二 .%> 

插/ V ,但是代碚 

<% • • • %> 

插>的人 


<li><a href="/ads/<%= ad.id %>"><%» ad.name %></a></li> 
<% end %> 


-味本歿的宋 4: •这夸 


这段代码在值需要被插人的地方为循环代码和表达式使用脚本 
段。让我们看看如果使用脚本段来循环遍历 @ads 数组，索引页 
面看起来是什么样的。 
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添 in 脚本段 

在循环的每次故行中，员面 
郄会生成一个键捿 

这是你将要在 index • html. erb 楱 板中使用的代 
叭： 


< html > - 
< head > : 

< title>All Ads </ title > 

</ head > 

< body > 

< hl>Ail Ads </ hl > 

< ul > 

<% for ad in @ads .%> 

< li><a href -"/ ads /<%= : ad.id %>”><%= ad 
<% end %> 

</ ul > . … 

</- body > 

- . . • 

</ html > 

• ■ . • • • . • 




5 RaUs 处坪这 个模 板的 时候，文件顶端 W 底部的 HTML 就适你想^输 
出。心趣的部分适 fr : 贞面的屮 1 擗环的好次执行郎幺生成：个指肉对 
应的广告负面的 HTML 链接。 


来 


达样你/ 



name %></ a ></ li > 
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法成的 HTM 厶是什么样的呢？ 

假设数据库中只有这3条 广告。 

这意味肴控制器将产生一个 @ ads 数组，其中包含 
广3个模型对象。当页面校板对这个 @ ads 数组做循 
环遍 历的 时候，它应该牛成像下面这样的 HTML : 



符起来这会为数据库中所冇的广&牛.成迠够多的 HTML , 如果数据库中创违 
了更多的广告，就会产生一个更人的 @ ads 数组，而页而模板就会生成.段史 
长的 HTML 。 

这就足整个过程。现在路由已经创达了，控制器的动怍代码也 S 完了，而且 
index . html . erb 校板也到位了，是时候运行整个代码 f 。 
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试驾 




复习要点 - 

_ 你可以显示来自单一记录的数据。 ■ 现在你有能力写很多很多只读应用了 ! 

■ 你可以显示一个数据表中所有记录的 
数据 
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你收到？ Mebay 芨来的 
一封 E-mail. 

网站的功能现在完全符合最初的要求了。所有人都很 
高兴。接肴，在网站要启用的那天早上，你收到了一封 
E - mail : 


朋友！ 

你把这个网站做得很不错！我们对你能完全按照 
我们的要求实现它感到很满意。我们早就听说所 
有的 Rails 应用看起来和运行起来都是一个样的！ 

另外，这儿有一份对这个网站外观的设计。我们 
相信这将是应用的最终外观了，但是如果还有什 
么改动，我们后面会再发给你的。 

再次感谢你的所有辛勤工 作！： -) 


这份 E - mail 里有一个怍本网页，还附带了一系列样式表和 
W 像。史改应川的外观应该并不难，是不是？ 


你可以直接修改页面模板从而让它们看起来和设计者提 
供的样本网页相似。这么做会有什么问题呢？ 



T 栽迗个/ 


从这儿下载样式表和 图像： 

www.headfirstlabs.com/books/hfrails 
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设计：:的应用样式 


fs 是有两个贞 i 糢板……我们要修改 
毎个糢板的代码吗？ 


我 f f ] 有两个页而镆板，所以如果你 H 是在两个揆板屮修改 H T M L 来 
适应 McBay 的样本 M 面，你就需要重 复编写代码。 这 样会有 人问题 
吗？毕竞， MeBayN 站只有两种 N «。 这并不算太糟糕，足吗？ 

问题 ft f •应用吋能会随猗时间不断扩展，会畨要越来越多的功能和 
页面模板。还记得那句设计可能变化的评注吗?你复制 的外规 越多, 
你需要维护相冋 HTML 的地//就越多。一段时间之 G ， 这个应用就 
会很难维护。 

所以答案是什么呢？好吧，很砬然答案是消除乘复。大多数 N 
站对它们的大多数! K 向都有 fc 准化的外观。它们使用标准的 tf - 板 
( boilerplate ) HTML ^ 闹毎 itt 的主要内容。 

所以你需要个定义超 级模板 （ super - template ) 的 方法：•个控 
制一组其他校 fe 外观的单一模板。 


HailsJ^flU : DRY 
>要重箕你令己 


*1 VS*N 



布爲定义？ 一组页面棧板的标准外难 

幸运的是，这样的超级模板存在干 Rails 屮，它被称为布局 
( layout ) 。布局为厲于特定模型的所有模板定义了 HTML 包装 
器 ( wrapper ). 

让我们看看它是如何与新设计一起工作的。 
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t 超级糢板 
代码上菜 


这是由设计者提供的转换成布局之后的 HTML 页面范例。 


<!DOCTYPE html PUBLIC "-"W3C//DTD XHTML 1.0 Transitional//EN" 

"http: / /www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
<html xmlns=' f ht.tp: / /www.w3 .org/199-9/xhtml'^ 

<head> 

<title>Ads : <%= controller.action_name %></title> 

<%= stylesheet_link^tag 'default.css' %> 

</head> ~ 

<body> 

<div id= w wrapper M > 

<div id= n header M > 

<div> 

<hl>MeBay</hl> 

<ul id** M nav w > 

<li><a href= M /ads/ M >AIl Ads</a></li> 

</ul> 

</div> 

</div> 

<div id="content"> 

<%= yield %> 

</div> 

<div id="clearfooter' , x/div> 

</div> 

<div id="footer M ></div> 

</body> 

</html> 


你需要把这段代码保存到如下的正确 位罝： 
app/views/layout.s/ads .html .erb 
这个名字告诉 Rails 把这个布局佐用到所 有厲于 atl 模型的页面模板。 

我们已经放人了一•些表达式来指定样式表并裉据当前控制器的名字来賦 P 页面杯题。但是 
更重要的是，布局要包含这个 标签： 

<%- yield %> 


i 磨笔上阵 




把当前页面模板的输出插入到布局中有没有问题呢？如果有， 
请写在下面。 
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修改你的模板。 

- - •.… - - *■ •- - - —- ■•- •. . :' 

~" ： ' - __ , "i- ， 1 -. - . - . - - . - -• ■•- •- - _ ― • - • -- 

. -- ： --—— • • - .. - .* * .. ■- -- . _ -m. • • : -r .. 

- -.. .•- -圓- •- - _ •产 • - - -- - - - ■•■： ■: ：r. . - < - . - - - . _. ••• ... - . • - - .•- 

笔上晬 .— -- - -- , 

解答 把当前页面模板的输出插入到布局中有没有问题呢？如果有，请 

写在下面。 

网吞榉板芭含5大多内容一古 fH 也 S 级忽含 r 鰣穷的 HTML 碑扳： 



你需要从克®模板中刪賒样板 


看一下 现有的 index . html . erb 。 他 
已经包含了 HTML 样板 元素， 比如 
<head>^ll<title >： 

… )： 


但是现在已经有一个提供样板的布局了 
仅仅诚示主耍的 K 面内容： 


你志 要战减这个模 板使 它们 


<hl>All Ads</hl> 

<ul> 

<% for ad in @ads %> 

<u >< a href -/ads/<%- a d.id %>-><%= ^' narae %></3></U 

<% end %> 

</ul> 



编辑 index . html.erb 和 
show . html . erb 文件以删 
除样板 HTML 。 
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但是 MePay 芨送过来的新的靜态 
沟容怎么4 ? 

到 H 前为止，你仅仅生成丫 Rails 应用中的动态内容。 确实 
大多数内容由页面模板输出。但是当你指定网站的外观时， 
你经常盂要类似于样式表.阁像和 JavaScript 这样的静态内 
容。但是你如何在应用中包含静态内容呢? 

Rails 特怠为静态义件保留 r -个文件央。它被称沁 public。 


当你创建应用时， Railstl 经把一哼文件放到 : Tpublic 文忭 
夹屮。还记得你第一次启动 Rails 应用看到的情况吗？ 
与标准的欢迎页面相关的文件都被放在 public 文件夹中。 

大多数 Rails 应用把它们的围像、样式表和 JavaScript 分 
别存储在 public / images 、 public/style sheets 和 
public / javascripts + 0 

一旦你把电子邮件里额外的图像和样式表保存下来之后， 
我们就可以继续了。 
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mebay 看起来很棒 



打开浏览器， 指向: 



http://localhost : 3000/ads 
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根据揭示神秘词语的线索在网格中填写答案。 

针对神秘词语的 线索： 

你需要手工创建一个应用而不是使用支架的原因。 






线案： 

1. <% @ what . am __ i ? %> 

2. 你可以为此使刚•个页面模板 

3. 把数据库中的数据转化成 Ruby 对象 

4. <%== @what . am _ i ? %> 

5 . 吋以从揆型中发送数据给视阁 

6. 使用 rake db : 史新数 据结构 . 

7. 如果你创达一个 简电的 应用，你 "r 能不需要它 

8. 从数据库屮读取对象 

9. 每个路由都有•个请求 . 

10. 包含很多对象的对象 
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破解 


无矣猓闪柃解著 

根据褐示神秘 W 语的线索在 M 格中填写答案。 

针对神秘词语的 线索： 

你需要 fl : 创建一个应用 iffi 不是使川支架的原因。 


砷秘 •:（ •:备 



1 r^iraranraRTiriirFi 
2 MHfiH 


3000 囹回 


4 fellx 

tF>||^||e||5||5||.||o|| n| 

5 LcJloJlNjlrllslL2JLs= 

IUI elM 


6 M 

1 i II dl 记 II All Tile 

7 3 

_c 

^ramraraf^i 

8 F 

1 

Nl^llell^l 

9 

P>| A 

r 


io IaJI^I 

r| a 

Y 


绞索 

1. <% @what .am i ? %> 


2. 你以为此使用一个页 面模板 

3. 把数据库中的数据$4化成 Ruby 对象 

4. <%= @ what . am _ i ? %> 

5. "了以从模型中发送数据给视阁 

6. 使用 rake db : 更新数椐结构 . 

7. 如果你创逢一个简单的应用,你可能不需要它 

8. 从数据库中读取对象 

9. 每个路由都冇一个请求 . 

10. 包含很多对象的对象 
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你的 RailsX 與箱中的工爯 

你已经把第2章收入囊中了.现在你已经将手 
工创建只读应用的能力加入了你的工具箱。 


X S 

你芍以用下$的命今來'主成一个榉 
m by sorlpt / ge ^ ratt 』丄.. 

以 5 /主威一个控 制器： 

r u.by scrlpt/gc^^^ 

i^w.by J |2 

— 1 一 釦菜 — 个 Ruby 盎钼， 

第一个尤 #4 ii 样碥定的： 
^ ij_arraLjloJ 

仿 .^ Jk / •这样 綈坏邊 总所有的无棄 
f or titv^tv ^： Iv^ kvty «rr«y 
# stuff* with tUv^tv^Jt 
CmI 
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3 插入、更新和删除 


命 


本一切都在变化中+ 



变化无处不在 尤其对数据而言。 到 R 前为止，你已经看过如何 
通过支架來迅速搞定一个 Rails 应川，以及如 W 编写你自 d 的代码来展示 
数据库中的数据。但是，如果你希望用户能够 A 行编辑数据的话应该怎 
么做呢？如果支架无法实现你所期®的方式呢？你将会在本章学习到如 
何用你所希望的方式来 插入.更新和删除 数据。而当你这么做的时候， 
你就会更深人地了解 Rails 是如何运作的，而且你还能学到一点儿安全知 
识。 


这是新的一章 






mebay 在回归 

人们希望在线绛贴 新广告 

人们非常喜欢 MeBay 网站，但是有个 |"j 题。因为 Mebay 枳心人们过 
多地访问数据,所以卖方不得不电话告知 MeBay 他们要卖商品的 
信息，然后再等上-会儿让系统管理员为他们创建新的商品广告。 
由于冉广告的人数 任不断 增加，他们的等待时间也就彼不断拉长。 
这就洁致现在很多人把业务的而 投向其 他广告站点 r 。 

所以 MeBay 做 fU: 步。经过一游讨论，他们决定允许用户使用如 
下的仇曲在网站 I •.张貼他们自己的 广告： 
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你 B 经知道如何褡建一个发布数椐障 
中数椐的应用 



当前应用中的广告只有-条道可走。广告记录由模型从数据库中被读 
出，然后模切把 记 - jHt 化成广告对象，这些对象被控制器发送给视 
图。工作过程 如下： 


认 0W. 

ft 诜®令巧见 


s ^ w . M>uUrb 挣板试 
致冲 ad 2 I 冉龙 吻容五 
•子 硪一 个轲毋 



你现在的位置 ► 


105 







保存和读取数据 



保存数椐就係锿取数提的反尚 
那样工作 _ 

保存数据到数据库与从数据库中发布广告很像，除了它是反向运作。 
你并不需要一个陡冶广告的页面模板，相反，你需要一个提交广告 
的页面模板。你也不需要一个发送广告到页面的控制器方法，相反， 
你需要一个从页面读取数据并把数据变成对象的控制器方法。同样， 
你不需要 lh 模型读取记录并把它转化成对象，相反，你需要让模型 
把对象转化成数据库中的一条新 记录。 
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你 f 要一个用采桡交数椐的表单 
和一个保存此数椐的动作方 

你需要-个新页面模板来创建 HTML 表黾。由于它被用来输人新广 
吿，我1把这个模板你为 new . html • erb 。 



这个 “New ad ” 豇面需要在如 F 网址显 


而表中.将被提 交到: 


所以我们也畨 g 在 r ou te s • rb 里创逑一 
条新路由。 ^ 


ads 控制器里的 “ create ” 方法。 



控制器的方法需要根据表单中的数据创建一个广告横型对象.你 
能找出模型中的一个在表单中没有对应的域的厲性吗？为什么会 
这样呢？ 


缺少原因 


把 /ads/new 关联到 new.html.erb 文件的路由长什么样呢？关 
联 /ads/create 与 ads 控制器的 create 方法的路由呢？ 


你现在的位置 ► 
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id 是生成的 





控制器的方法需要根据表单中的数据创建一个广告模型对象„你 
能找出模型中的一个在表单中没有对应的域的氧性吗？为什么会 
这样呢？ 


缺少原因 

tA. 


把 /ads/new 关联到 ■new.html.erb 文件的路由 长什么 样呢？关 
联 /ads/create 与 ads 控制器的 crcau ： 方法的路由呢？ 


你的 config / routes . rb 文件的顶部看起来 W 该是这个样子: 



Rails 应用的许多组成部分之间都有很亲密的关系。 

中竟，模哲每含应用的数据，视图允作用户访问这苎数据*而榨制器 
提供 T 逻辑黏合削来把 W 迮接 起来。 


佴是表咿和投 蝥之 N 是斤 有矜特殊的*系呢？ 
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表单乌对象相兵吗？ 

除丫生成的 id 之外，衣黾中的域与广告对象的 wn 完全叫配。 



描述 ( description ) 域 将匹配 描述 ( description ) 属性, 
以此类推。 

如果模喂使用祕性的默认值来创建对象会怎样？生成 
表单中默认值的代码与模％代码是否略显重 k ? 

毕毚，表单中的数据婭由控制器来接收的，表单足否 
应该把这些域 fH 乍单独的值？还是所冇这些域的值应 
该被联系在一起，就像对象的 W 性那样？ 


Rails 在创建表单时能否利用表单域与模型对象之间的 
关系呢？ 


你现在的位置 ► 
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rails 根据模型创建表单 


Rails 能够劍建乌糢型对象相 
兵联的表单 


Rails 能够使用模兜对象来帮助创建表笮。这怠味荇两汁- 
事倩: 



name = M adfemail ] 




type= M text 


O 表 单域中 的值将被设®成 @ad 对染的 《 性中存 lift 的值。这 

对广告表单+会有太大的彩响.因为新广告是空 C ] 的。 

O 表黾域将被给 f 与模型对象中相应字段显式关联的名字. 

那么名字如何能把域 *5 对象关联起来呢? it 我们罾一下这 
对广告表单的意义《下面&为基干 Ad 对象 的丧单 生成的 
HTML： 


fcr 



姑 o 涊您 


比较一下表单域的名宇和它们的匹配属性. 
你认为 Rails 如何把这些表单数据传递给控 
制器， 

你有几页的时间来考虑这个问题 …… 


〜表单域的名字 

对象厲性 

ad[name] 

name 

ad[description] 

description 

ad 【 price 】 

price 

ad[seller_id] 

seller__id 

ad[email] 

email 

ad[img_url] 

img_url 
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m 


砉单对象代碚冰葙糍锈 

是时候编写 new.html.erb 页面模板了 。<% form _for %> 
标签被用来生成一个使用表单对象的表单 3 请使>下面的 
代码冰箱磁铁完成表单中的各个域 


••••• 我 


^ <hl>New ad</hl> 


表輩杉荃軚基-芟於色錡以不 1 
在 I 面沒用 =• 


<% form for(@ad, : url=>{ : action => 1 create'}) do IfI %> 
表掣 奸 <p><b>Name</b><br /><%= f. text 一 field : name %></p> 


表輩柃含歧祛交 iii 个劫 

- 


<p><b>Description</b><br /><%= %></p> 

<pxb>Price</b><br /><%* %> 

</p> 

<p><b>Seller</bxbr /><%= %></p> 

<p><b>Email</b><br /><%= %></p> 

<pxb>Img url</b><br /><%» %></p> 

<p><%= %></p> 

<% end %> 







[^^scriptionj 


f.submit 




f.text_field 




: img 一 url 


f.text field 
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表单冰箱贴 




砉单对象代碚冰筘糍铁解考 


是时候编写 new . html . erb 页面模板了 。 form _ for h 标签被用 
来生成一个使用表单对象的表单.请使用下面的代码冰箱磁铁完成 
表单中的各 个域： 


破称妗 Y 


<hl>New ad</hl> 

<% form_for(@ad, : url=>{ : action=>'create'\) do |f| %> 
<p><b>Name</b><br /><%= f. text_field : name %></^ 
<pxb>Description</b><br /><% 




f.text a 


Ij ^^scription I 


%X/P> 




<pxb>Price</bxbr /><%= .| f• text 一 field 

</p> 

<p><b>Seller</b><br /><%= 
<pxb>Email</b><br /><%= 

<p><b>Img url</b><br /><%= 

<p><%= 



f.text field 


f • text_field 


1 -email __ 

: img_url |></ p > 




f.submit 


<% end %> 


"Create" || 


></P> 


♦ 


问： 


鮮 ㈣ ms 

+/0 

达样你 / 将这段代码保存到名为叩卩/7^»#3/3€13/116讨.以1111. 

erb 的文中。 

沒存蠢 H 题 

有蠢网题； 



答： 


为什么表单中没有对应干 
id 、 updated_at 和 created_at 这些字写 id 会自动生成敫字，而 updated 
段的域？ at 和 created_at 域将被给予记录在 ft 


据库中被保存或者更新时的时间戮. 
这些域将由 Rails 自动填它们是我们的神奇字段，还记得吗？ 
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针对 “ new ” 表单的页面模板已经就绪，而它应该能够使用与我们生 
成 “ show ” 页而一样的方式来生成 HTML 。 同时我们还有一条准备 
好的路由，它把 “/ ads / ncw ” 路径与 “ new ” 模板关联起来。让我们 


看看如下 URL 能否生效吧: 


http://local host: 3000/ads/new 




Action Controller: Exception caught 


» j C ] [ ^ I 1 4- I ^ hnp://localliost:3000/ads/new 

_ _ _ : _ : __ 


RuntimeError in Ads#new 

Showing ads/new.html.erb where line #2 raised: 

Callad id Cor nil« which would miatax«nly be 4 -- if you really 

Extracted source (around line #2): 


object_id 


<hl>M«w ad</hl> 

<% f om_for< 9ad, t url»>{ t action a > ' create' } ) do |f| %> 

3t <p><b>Name</b><br /><%- f.text field t name %></p> 

4 1 <pxb>D«»cription</b><br /><%• f.text area tdescription %></p> 
5 1 <p><b>Pric«</b><br /><%* £.l«xt_fiald ipric* %> 


页镝出 # 3 / 


^ 汸奶® - 

这个表单页面有问题请者看这个生成的错误消 
息 . 你是否能够想出是什么错误？ 


你现在的位置 ► 
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nil 指空值 

@ ad 表簞对象还没笮被劲建 

问触由_对象服的。㈣ d 这样的变賊 

朱的称为 hi 酿，也就是空值。 ^Wad^VxS^n, ■ 
贈，它将■獅 ad. — ，@ 

ad.de script ion 这样的辑性。 

如果 @ad 不具有刪峨，糾奸柿响吗••’ 

没氏 会有影响！表单依赖千㈣对象.它需要访 
个^性来生成表申.中各个成的初始俱 • 喊作 •第一 1 
用时就返回了 nil， 这就汗致了错误。 

那么我们该如何避免这个问题呢？ 


SSIIrH ： 

Ofill 


@ad = nil 



<hl>New ad</hl> 


<% form_for( : ad, : url=>( : action=>'create'}) do If I %> 
<pxb>Name</b><br /><%= f . text_field rnamp %></ p > 一 
<pxb>Description</bxbr /><%= f.text 一 area : description %></p> 
<pxb>Price</b><br /><%= f. text_field : price %></p> 
<p><b>Seller</b><br /><%« f.text 一 field : seller ■ 一 id %></p> 
<pxb>Email</bxbr /><%= f. text__field : email %></p> 

<p><b>Img url</b><br /><%« f•text—field : img_url %></p> 

<p><%= f.submit "Create" %></p> 

<% end %> 


« 个皱的初诒 $ 黨 
$ (务 H 今邃 ad 上极 
S •圮的霣作 D 

<^ L . ii 軚异致了 

持復' 
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表单对象 f 要在表单钺显示之前 
创建好 

'■1 带有表承的页面被生成时，毎个表单域的初始值均来自千其关 
联对象的各个属性。 


你知遂这样会布什么问越吗？ 

M 题是，在表申.被牛成之前新的广告对象钛需要存 ft 。 当然.在 
用户完成广告的详细怡息之前，广告对象不会被保存到数据库 
中 W 即使如此， 这个 对象依然需要在页而模板被调闬之前先 
创建好。 



象 一一 任4它不用淨存到 
兹馮碡 t . ci 个对象保仅 
用皋 s ‘ ja 表# t 



你现在的位置 > 


115 




新方法 new 


表单广告对象将在控制器的 new 动 
作中狨创建 ~ ' 


O 表单广告对象需要在 new . html.erb page 模板运行前被创 

建。 

如果你在名为 new 的控制器方法中创违这个对象，它会在 
new . html . erb 模板被调用前先运行。 



浏览器请求 调用广告控制器 

http://mebay.com/ads/new 的 “new” 方法 . 


先峡用 


def new 


end 


O 那么你该如何 创建一 个新的广告对象呢？ 

Ad . new 返回一个新对象，你可以把它陚给 @ ad 变量。新 
对象不会自动保存到数据库中，你只需要 it 它存在干内 
存中，以便干生成 HTML 表单。 






def new 


@ad = Ad.new 


end 






O 


现在对象已经被賦给内存中的 @ ad 变里， new . html . 
erb 可以使用它来生成 / ads / newN 页中的 HTML 表单。 





new" 方法创建新对象 . 


new.html .erb 模板使用这个对象来生成网页 , 
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现在毎个贞面糢板都有一个 
西£的控制器方法 


现在广告控制器对干毎个页面模板文件都有一个相应的 
方法。 Rails 在从页曲揆板生成页面之前总是会先调用控 
制器的相应方法。 

控制器方法和奴向模板共同组成 r 动作。这就是为什么 
动作名字既出现在控制器方法的名字里，也出现在页面 
模板文件的名字里。 

动怍 由控御 粽方法扭页审 


app 


controllers 







| class AdControlle V ApplicationController 
def new 


new.html.erb 


show.html.erb 




ads_controller.rb 


index.html.erb 


现在，你的控制器能在 new . html . erb 页面模板运行之前 
先创逮新广告对象，是时候看看代码是否能够奏效了。 


你现在的位置 ► 
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mebay 在行动 



在控制器代码准备就绪后，是时候试试这个应用了 
用浏览器打开如下 URL : 

http://localhost: 3000/ads/new 

现在输人一些 数据： 


让我们 


» 


表单! ft 面显示得很完美，但是当你输人一些数据并提交表单后 ， Rails 
返 M 了一个错误来告诉你你还没有编写 create 动作。 create 动作？这个动 
作用来从表革接收广告信息。 


这个动作需要做些什么呢？它需要读取来自丁表单的数据件使用这些 
数据来创建一个新广告。 

但是表单发回给应用的是什么呢？ 
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表单不 会崖送锣对象，它崖 
送诊数椐 

这儿的表单是通过一个 Ad 对象来生成的。但是当表笮被提交 
时，表单会把什么发送回服务器呢？ 

因为表弟使用 HTTP， 所以它无法通过网络发送表单对象。它 
只能发送数据。 

侄是数掩是如何组织的喂？ 


回想一下路由是如 H 工作的。当某个请求到达时， Rails 把请求 
的 i 羊细信息发送给路由系统，这样就会把钋对动作和控制器的 
值插人到一个名为 paramsU..】 的数据结构中。 


params[...] 数据结构并不仅仅为路由而创途。它也可以被用 

于 ㈣ 任何通过 Web 表单提交给 应用的数据。 ^- - 作 W 个朗狀 S . •料柳 

表单中的域以 名字： ad 被 i 己请在 paramo...] 表里。 ： ad 变罡 

的值事实上是另一个具冇多个值的农，这个表把域的名字映射 Array) 

到域的 fft: 


参數••呤杳’•表. 



⑽.■刼 •_ 






Name 

Value 

: controller 

'ads' 

: action 

'create' 

: ad 

Name 

Value 


/ 

: name 

Leather boot ^ - 

- 

( 

: descriDtion 

Suit oerson with left foot 


.ad 夸 ft 的 

: price 

1.0 


呓 4 另一个 

_ : seller id 

242 


: email 

bert§hotmail.com 


: ima url 

httD ： //www.iavaranch.com/imaaes/boot.aif 







,st：ac*l ^ 


但是当控制器接收到数据时，它会发生什么变化呢？我们 
应该如何使用这些数据呢？ 


你现在的位置 ► 
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从数据到对象 


Railsf 要在数椐狨保存之前拕数 
椐转化成对象 

Rails 弓数据库交流时只能使用对象，所以在某个广告被保存进数 
据库之前，你需要找到•神方法来把表学.数椐转化成一个 Ad 对象。 


糢型能够根椐原始的表簞数鴂创建对象 



Ad . new 方法也能够被哈希表调用，哈希表中的值被用来初始化新 
Ad 对象的厲性。表申.数据正好包含在一个哈希对 象里： 


@ad = Ad.new(params[ : ad]) 
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手拕手敎你控制器的 create 

O Rails 使用 params [_. .】哈希表发送表单数据给控制器。 

params 



create 方法 


def create 

• • . pSTellTlS ["•] 

end 


O 控制器通过査看 params [: ad ] 来读取原始表单数据。然后它 

把这个值发给 Ad . new (...) 来构造一个新的 Ad 对象。 




end 


你现在的位置 ► 
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保存记录 

控制器 f 要保存 is 彔 

把表中.数据转化成对象的原因是让你能够保存它。 

你该怎么做呢？使用 

@ad.save 

当我们在揆型对象 h 调用 save 时， Rails 检査対象的属性并生成一 
个 SQLM & 语句来电新数 据库： 



在 save 被加人之后，控制器的 create 方法就完 整了: 
def create 

@ad=Ad.new (params [:ad]) 

@ad.save 

end 
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更新你的控制器的 create // 法，然后测试这个吏新过的广告创达表 
羊。当你点击 “ Create ” 按钮时，返回了一个错误页面表明缺少某 
个 模板： 


^ O O Action Controller: Exception caught 

v: - Q- 

Template is missing 

Missing template ads/create.html.erb in view path 
/Users/davidg/mebay/app/views 



2. 你需要怎么做才能解决这个新错误？ 


] | C~| - It^jj ^ hup://localhost:3000/ads/create 
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每个请求都需要一个响应 




笔上阵 
^解答 


.你创建的记录保存了吗^ 
數麴铍 係存.扣广苦破舍 jit 。 


2. 你需要怎么做才能解决这个新错误？ 

I 含 J 燶一个 Grcflte.libKl.erb 炙面。 


Rails 显示错误是哆为它无法为你 
的清求生成喰应 

HTTP 使用成对的请求和响应来工作。对干每个请求，都应该有个 
响应。 

当控制器的 create 方法执行完后，一条新记录被成功创途。然后 
Rails 需要生成一个响应页面，所以它査看能够匹配当前动作的页 
面模板。当前动作是 create , 所以它査找 create . html . erb 。 
但是这个模板并不存在！ 



所以我们需要编写 create . html . erb 页面模板。但是这个模板应该 
包含什么内容呢？ 
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■ 口 

GQ 


I 建一个 create . html . erb 文件来告知用户记录已经被创建并提供 
个指向新记录的 链接。 


?000 


Title 
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给你响应.女士 





你应该创建一个 create , html . erb 文件来确认记录已经被创建并提 
供一个指向新记录的链接。 



雨杉.跆扣斿知軔广告的链尨杖巧- 


<V\±>A.d created !< /hi > 崎命於餐入的 


vUw youcr ad <a href = "/ads/<^5 = @ad.Ld < j^>> n >V\trt</a> 



忠 t: w =r 卿 


久、 i 妒1广告的珞 
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现在你的0^316.111：1111.61：131^断模板11经就绪，是时候 
试试这个应用了。 



Pncaci.O 


SMf M:M 


很快用户就张贴了几十个新广告。但是尽 
管它很受欢迎，有些人还是有些小意见… • 


你现在的位蜃 ► 
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重用现有的模板 


不要劍建新页 i , 使用规冇页 i 

用户并不想看到由 create, html.erb 生成的额外的 
确 U 页面。他们只想 食接到 达他们的广告 K 面。那么 
你该怎么做呢？ 


你可以编辨 create .html .erbij( 面校板以 U ： 它新 
广告的作细信息，对吗？中竞， @ad 变 S 吋以被贞面 
揆板访问， i«R 它包含丫新广 ft • 的所有佶息。 


m 这是一个楢糕的想法。沁什么呢？因为这就总味荇 
create. ht m 1 . e rb 页面将 $ 得跟你 用来显 示甸个广告 
的 show, ht m 1 • e r b 奴 _ 投版一模 - 样。 


接 N 疒务 



疼！ • 丫 琢的 •不金 


t iCt-t) 6 


1 


这足重复劳动 . 而且这意味 C 将来你 ff 史多的代码 
需要维护。如裝控制器中的 c r e a t e 动作能够显示 
show.html.erb 页面的话会史奸。 





show.html.erb 


祛入卢苦 


0 - 



横型 


控制器 







S 斤 • r 苦 
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show.html. erb 





插入.更新和劚除 


侄是 控制器劫作如何才能显示 
另一个 动作的页 i 哝？ 

控制器方法弓模板 JtM 组成一个动作。在前面你所#到的所 
存例子里，控制器方法和页面模板都足被特定动作独占使 m 
的。 

这也足为什么控制器方法和页面模板都在一定程度 h 包含 r 
动作 名字。 、你 执行 Show 动作时，你使用了广告控制器里 
的 show 方法和 show, html • erbjJi |fii 校板。 

我们依然 希％使用 控制器 * 法和页面模板来完成这个动作， 
fH . id 现在我们需要能够选择控制器方法所调 IH 的豇 面模板。 


拉剌粽动作由拉 
剌粽方法！ 菸夜 
梂犋祖成 "T 



http://mebay.eom/ads/1 0 


我们需要为控制器找到一种方式来表明输出位于一 个不同的 
URL 。 """ 
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重定向到另一个 URL 


重定尚使控制器能够指定盎示彿个 

重定向 （ redirect ) 是 Rails 返回给浏览器的一种特殊响应。 

它告诉浏览器到一个不同的 URL 去获取输出。所以，即使浏 
览器把表中.数据发给 / ads / create , 重定向也会 ih 浏览器来 
到 ads /17 (例 如，假设 17 足新广告的 id 号）。 
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插入、更新 I 删除 




磨笔上阵 


redirect 命 令将把浏览器转到新广告的 URL 请在下面虚 
线处填写出该 URL : 

def create 

@ad = Ad.new(params[ : ad]) 

@ad.save 

redirect_to "/ ./# (.}" 

end 


f 没存蠢码蕺 . 

々苕蠢舰 


你说重定向是一种特殊的响 
_ 应。它真的是响应吗？ 

是的< 有几种类型的响应。 
通常的响应包含浏莧器需要显示的信 
息，但是其他呤应，比如重定向，包 
含一些针吋浏見器的特殊拍 4' 重定 
向犹是一个特味指令.它要求浏 t 器 
跳转到一个不同的 URL 

那么当 一 个浏览器被重定向 
时，地址栏中的 URL 会变化吗？ 

你锖对了。即使浏見器清求 
一个特定的 URL , 地址栏最终也会显 
| 示浏見器结束本次请求时所在的 U R L 

• 重定向的代码出错会导致浏 
览器陷入重定向的死循环吗？ 

不可能.因为洌 3 L 器对重定 
向的次啟有限制。如果你的代码不断 
地发送重定向，洌瓦器将会不对熥， 
它会停止重定向动作并显示错误消息 


问： 如果我把一个对象赋给@&1 
然后重定向到不同的 URL . 新 URL 能 
够看到我的 @ ad 对象吗？ 

答： 问得好！不行，它做不到 
一旦你重定向到另一个 URL . 你将无 
法在新地址访问在前一个地址由控制 
器睐予 的变量 

我能够重定向到一个不在我 
应用中的地址吗？ 

当然 u 重定向只是闷单的要 
求浏見器跳转到一个指定的 URL 。 如 
果你要浏見器重定向到一个卟部网 
站.測笟器就会这么做。 

我什么时候需要重定向？ 

当你需要重用显示信息的页 
面时你可能就要做重定向这也是你 
在 MeBay 应用 i 使用重定向的希因 
不过在你修改数据库时也可以使用重 
定向。 


| P ). § 为什么呢？ 

较好的做法是把动作分成两 
块： 一诀是更新动作，另一块是显示动 
作更新动作将会修改抆据库中的内 
客然后重定向到盞示动作 u 这样，如 
果有人榆人一条记浪，然后在下一个 
页面点击利新•^铉.他们将只能刷新一 
个显示页面.而不会重新插入记录。 

1^1 • 重定向字符串中的 #{} 是什么 
意思？ 

Ruby 字符串可以包含 Ruby 表 
达式 就涞 Ruby 名字一徉,，把表达 
式放到 #{ } 中会让 Ruby 自动把表达式 
替換成相应的值 

^ • params [...] 看起来有点像 
数组 . 它是数组吗？ 

: params 被设计成一钟“关联 
致组 （associative array )". 或者 
叫“哈希 （ hash ) ” 。 哈希是一种特 
沬的敫组，它能够使用其他东西而不 
仅忟是敫字来作索 
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创建，读取.然后是更新 




笔上阵 
^蘇答 


redirect 命令将把浏览器转到新广告的 URL .. 请在下面虚线 
处填写出该 URL 


def create 

@ad = Ad. new (parauns [: ad]) 
@ad.save 

redirect 一 to H / /# { 

end 





请史新你的控制器。现在是时候来 打 行我们的歌 
定向适杏把 m 户送到他们新创边的广& 1：： 



当新广告被创建时，浏览器自动跳转到了新页面。 
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插入.更新和删除 



有些用户他们的广告创 迮过程 屮输错 r 内容，他们希嗜能够修改 
他们的广告。所以他们不仅仅需要显示和创达表申.。现住用户还需 
要能够编辑他们的广告。 




这就意味着系统需要同时支持更新和插入。这会很难实现吗？ 
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更 新记录 

E 新广告就係釗建广 t 一 
样 …… R 有一点小 g 别 

即使系统 h 前还不能够编辑广告， m 添加编辑功能会需要很多 
工作吗？ 

想一下把广告插人到系统中的步骤： 

O 新的空白广告对象被创建，然 后被用 来牛.成广告输 

入表单。 

O 表申被 发送给用户，用 p 更新 各个域 的值汴 提交表 
单给应用。 

O 数据域被转化成 Ad 对象，它敁保存到数据库中。 
o 用户被带到一个显示新广告的 m 。 



1. 创建空白 
广告 


2. 煩值 3. 保存广告 4. 显示 


假定你要修改一个广&。你希頦通过什么忭的形式來修改呢？也 
I 午足•个带行广告信息的 表承， 你 "r 以 t 新提交这个衣单来保存 
所冇的改动，这和你创违广告时的步猓是一致的。 it 我们柬仔细 
看一下修改的步骤…“ • 
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你 f 要找到一个广告而不是釗建 
它，你新迖个广告而 
保存它 

当有人编辑广告时，他们使用与前面一样的表单。用户将会修 
改广告信息并保存数据。那么修改步骤和创建步骤有多相似 
呢？ 


o 从数据库屮读取现有的广告，这个广告被用来生成 

修改表铯。 

o 表单被发给用户，用户更新各个域的值汴提交表黾 

给应用。 

O 数据域被转化成 Ad 对象，它被用來更新数据库。 


o 用户被带到一个显示更新过的广告的页面。 



1. 读取广告 2. 修改各个域 3. 更新广告 4. 显示 


你可以看到这两种操作的步骤雇本没有什么差别。在创建过程 
中，新的广告对象被创建 )) ••保存到数据库中。在修改过程中， 
一个 现存的 广告被读取、修改然后保存到数据库中。 

所以你只需要关注这两种操作的差异之处就行了。你需要记录 
类似十修改步骤里的广告 id 号这样的数据。 


你是否能够把这样的编辑功能添加到应用中呢？ 
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变化是好事 


m 




在每个 show 页面添加一个指向某个 URL (比如 ads / lVedit ) 的编辑链接.该 
URL 会调用一个新的 edit 动作来查找广告并显示到表单中。 






表单将把数据提交到一个位于如 / ads /17/ update 这样的 URL 的 update 动作。请创建关联/ 
ads /17/ edit 到编辑页面和关联 / ads /17/ update 到 update 动作的路由.请确保广告 id 号 
在这两种情况下都被保存在名为： id 的请求参数里。 
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时代正在改变 __ 



p : 瓷确 
t 籾答 


你的任务是在每个 show 页面添加一个指向某个 URL { 比如 ads /17/ editl 的编辑链 




表单将把数据提交到一个位于如 / ads / l 7 / update 这样的 URL 的 update 动作你需要创建 
关联 / ads /17/ edit 到编辑页面和关联 / ads /17/ update 到 update 动作的路由.诮确保广告 
id 号在这两种情况下都被保存在 名为： id 的请求参数里 


/adi/ild/cdlt', : co»vtrolUr= > ' ads ', : actio^= > edit" ^ 

... … 保邙龙 ii 签路由播入 

. , . , • /〆 ^ ^ ^^^ftg/routci.rb 

M^flp.&oiA^ucct / ad^/ild/update', : c^ovJtrolltr= > . ads ’， '.action- > u-^datt' 迂作中的與的於由 p 節吞 
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插入，更新和删除 


你的任务是创建新的页面模板 edit . html 来允许用户编辑广告信息，它很像 
new . html . erb . 但它会在页面头部显示广告名称并使用 update 动作而不是 create 动 
作 


forv^. for(@ad/.uri= > {: actlo^= > w.pdfltc'}) do |f| 

<-pXb>N«>vi€</b><br/><^= : 多 ></p> 

<*p><b>Descri|itio^</b> <br/><^ 5 = f.ttKt_ana : dcscn-ptlokv 

$></?> 

<-p><b>T > ncc</b><br/><^= f.ttxt ^rlce 多 >< 今 > 
<|>><b>Scll€r</b><br/><^= : &clitr_ui ^></p> 

<p><b>e^vtflll</b><br/><^= f.te^t_fuld ^></p> 

<p><b>lkvt0 u.rt</b><br/> <°^>= : im0_wri 茨 ></p> 

< ? >< ? == f . utpoifltr ^></ p > 


你需要广告控制器里的两个动作方法.一个为编辑表单提供数据.另一个更新广告到 
数据库中。 edit 方法将提供数据给编辑表单.而 update 方法将更新数据库。考虑到 




@ad. update_attributes .... ^ 

—将把象更新到数据库中.你编写了如下代码来实现广告控制器的 edit 

和 up date 方法 


c^cftcfit dcf M.pdat€ 

你 d = Ad.-ft^d (parflM^s[ ： Lc<l) ^ad = Ad.fivui 

队沒 . @ad.u.jidatg_flttiabu.tcs (pcirOMs[ ： ctci]) 

rcdlrcc-t^ "/«df/#{@fld .Id}* 
e\^d 
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试驾 





edit . html . erfc 页面模板已经就绪，路由和广告控制器中的额 
外方法也准 ft 好 r 。 所以现在是时候测试新的编辑功能了： 






一 家！ 入妇该玆苟 

击 ~w ,拉往 


Editing Leather t 


DlPUmwi.Miy' dl^Mt 


b n 




卖家的 疒著崦 定錡斿 


All Ads 


卖家从索乡 If ) 表 中逢择 
的 们的广苦 


代码已经写好.编辑功能也挺好用的。那么我们的工作就完成 
7 ……对吗？ 
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插入、更新和删除 
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安全 


照制对某个功能的访问 

现在应用能 够创达 汴更新数据。但是这意味着任何人都能 
创违和更新所有的广告。这是-个问题。 

MeBay 的所有者洽嚷任 H 人都能创违广告，但是他们不希 
望广告在张贴后能被其他任何人修改。 



防止所有人都能修改广告的方法之一就是使用用户名和密码 
来保护更新功能。 

MeBay 员工决定只有系统管理员可以修改广告。所以他们希 
望新的更新功能通过管理员的用户名和密码被保护起來。 

幸运的是，在 Rails 里很容易就可以实现安全控制。我们将使 
用一种名叫 HTTP 认证 （HTTP Authentication ) 的 Web 安全 
机制。这种安全机制在有人试图进人网站的安全 K 域时会弹 
出一个对话框要求输入用户名和密码。 
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絮彔 ( Sign - in ) 
g 代碚上茅 

你需要在广告控制器中加人两段代码来把 S 录安全机制 （login 
security ) 添加到应 用中： 检奔用户名和密码的 login 方法，以及 
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阻止 • 



现在安全代码 l ! ■经就绪，蛙时候打开你的浏览器来试试编辑一个 
广 告了。 




现在只有那些知道 管理&1 用户名和密码的人木能在网站 . h 编辑广 
告，其他人不能得到编辑 页面 和更新功能。 


把页面和史新功能邯保护起来很重要吗？编辑页面应该被限制 
访问.以防止用户+小心进人这个页面而把时 W 浪费在输人数据 
上。+仅如此， *： 新功能（实际更新数据库的代码）也志要被保 
护起来，这抒" I 以阻止黑客不迪过编辑表单来直接访问它。 

你已经手工搭建了一个能够创建.读取和更新广告的系统，并且 
这个系统是安全的！ 



♦r S 
-'4 料 

A 

A 

:tt.e 

"2 名 
f 尸癯 
备 .f 达 
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插入.更斩和删除 


• 但是现在坩广告 f 要删轉 

网站 iE 常运行中，所冇的功能都运作得很好， 佴足 没过多久就发 
现了 -个 问题: 即使商品被卖出，广告还在那儿。 



MeBay 能够 使 / T ] 他们 fi 己的数据输入系统來刪除 M 站屮 
的广告，但是他们被修改功能的简易性打动了，他们希 
望你给网站添加一个_除功能。 

这个功能 H 能由 MeBay 苷理员使用，所以他们 ㈣ 样忠要 
那个应用至修改功能的安全机制。 m 时，山 r 有大设的 
垃圾广告和-些诈骗广隹■被张貼到网站 h , 他们希望在 
索引*面上就能获得刪除功能。这样他们较轻一点就能 
刪除不合适的内容。 


让我们看一下我们要做的事情 
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创建、读取.更新和删除 





更新 index . html . erb ， 在每个广告之后添加一个删除链接。如果广告的 id 
那么链接就指向 / ads /17/ delete 。 


17 , 


hl>All Ads</hl> 
<ul> 


<% for ad in @ads %> 


<li><a href-"/ads/<% is ad.id %>"><%= ad.name %></a> 

(<a href*" M >Delete</a>]</li> 


<1 end %> 
</ul> 


乂 4 

扣⑽’ 


创建一条路由.把类似于 / ads /17/ delete 这样的路径与广告控制器里名为 destroy 
的动作关联起来。记得把 id 号记录成请求参数 = 
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插入、更新和删除 


完成广告控制器的代码来删除一个广告并让用户的浏览器返回到显示所有剩余广告的 
索引页面为了实现它.你需要使用一个我们还没有见过的方法一 " destroy 方法 3 
这会从数据库中删除一个广告对象 


class AdController < ApplicationController 

before 一 filter : check_logged_in , :only => [ : edit, : update, .J 

def new 

@ad * Ad.new 
end 

def create 

@ad ■ Ad.new(params [: ad]) 

@ad.save 

redirect_to w /ads/#{@ad.id}" 
end 

def edit 

@ad = Ad.find(params [: id]) 
end 

def update 

@ad = Ad.find(params f : id]) 

@ad.updace_attributes<params [: ad]) 
redirect_to M /ads/# {@ad. idP' 
end 

def show 

@ad = Ad.find(params[ : id]) 
end 

def index 

@ads = Ad.find( : all) 
end 


二 : ：: ; 斡騎挪 


private 

def check 一 logged 一 in 

authenticate 一 or_request_with_http 一 basic("Ads"} do I username, passwordi 
username == "admin" && password == M t4k3th3r3dpill M 
end 
end 

end 
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这部分有些难度 





你应该已经更新了 index.html.erb, 在每个广告之后添加了一个删除链接。 

恤诗 坏悉® i f r 苦{|过 

<hl>All Ads</hl> X $ 下乘 
<ul> 

<% for ad in @ads %> 

<li><a href ss *'/ads/<%= ad.id %>"><%= ad.name %></a> 

[<a href= H /«?!&/<?=. n >Delete</a>]</li> 
<% end %> 

</ul> 


你也应该创建了一条路由.把类似于 /ads/17/delet:e 这样的路径与 ads 控制器里名为 
destroy 的动作关联起来。你把 id 号记录成请求参数。 


'/ctdi/ ： ld/dtiett\ ： cov^ro\itr=> ads', ： acXlov^=> destroy* 


/ aas / ^" 每由:以含耷 




f it^r ^ ..触印 y 
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在这儿你的任务是完成广告控制器的代码来删除一个广告并让用户的浏览器返回到显 
示所有剩余广告的索引 页面： 


class AdController < ApplicationController 

before_filter : check_logged_in, :only => [ : edit, : update, ..] 

def new 

@ad = Ad.new 
end 

def create t 

@ad - Ad.new(paramst :ad]) 扣，试 ® 獅咎一銮 —_ 

Sad.save ^ 

redirect_to w /ads/#{@ad.id)*' 
end 

def edit 

@ad * Ad.find(params[ : id]) 
end 

def update 

@ad = Ad.find<params 【： idj} 

@ad.update_attributes(params[ : ad)) 
redirect 一 to "/ads/#{@ad.id}" 
end 

def show 

@ad - Ad.find(paramsl : id]) 
end 

def index 

@ads = Ad.find( : all) 

(®arf = Ad.^^(^aravwsl ： id\) .. 

@ad.destroy _ 

redirect_to /flds/' 、 . 然后你禽 蜃将别 笵器 

. 重金余史索？ 1 否蛋。 

private 


def check 一 logged 一 in 

authenticateorrequest—with 一 http—basic("Ads” > do |username, password| 
username ==* "admin" && password == ,, t4k3th3r3dpill" 
end 
end 

end 
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再次试驾 



destroy 动作意味 昝用户吋以轻 轻一点就能从 M 站上 刪除 广告。他使 
用与修改功能•样的安全机制，所以在任 M 人刪除 广告之前，他们必 
须首先证明他们是管理员。 



A Ads 


记录•攻 蜊祅 


Typewriter [DeLgtei 
FooU3«U (Delete] 

Wootphead ( CXtfte ) 

Desis ICcitic) 

Pgpr ttiflgjn [Qflelfi] 

Acote Newton tDctetel 

SJnckirXS [Dtiete] 

Ediel IPetete) 

Dwnood [Setfie] 
leaihgf boot fPrtetel 
Days of Our Liv«' [Crate 


我 D 埒 s 铨达 


现在系统可以刪除广告了，你的应用能够 
执行所存基本的 CRUD 操作： 


工嫩 

〆 饰遠广吿办瞻咖 

祿取广告 (Reed ocb ) 

它们） 不 

g 薪广吿 
^刪隊广吿 


- 丨 ■廬 v.l aLL. 


«1 it M / 1 — ■■ /^Jh. 


写自己的代码呢？ 
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插入.更新和删除 


g 编写代码玎认让你实规比支 
¥1多的 功能 

你可以选择实现哪些功能。 

你 吋以添 加额外的类似于安全帆制的功能。 

现住 你理解了如何编写代码来插人、更新 和刪除 数据，你将 
能够修 改支架 生成的代码。 



你现在的位置 ► 
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rails 工具箱 


你的 Rails 工異箱中的工具 

y 你己经把第 3 章收入瀵中了，现在你已经将手 
/ 工创建应用来插入、更新和删除数据的能力加 
入了你的工具箱。 


X S 

@ ad . save 係存一个铗型的象 
@Bd.ucpdnte_nttrlbntcs i 韌一个機免 

rcdlrtc-tjoil: 控剌器把 ‘ 刘铱器榮至 ’） 

一个 不同的 

http 八芍以毫不费力 

祕-；^加在食机制 _ . ^ 1 - 

_心 cby 工^ 


paraw^s […] 署一个 啥每，达軚译 
一个使 用名字 来索幻 的袅钼 

觀咐丄 •㈣ 
fj. J°„" } 秸够把表 ii 式#入到 

^ 14 +1=#{1+1} " ^ 
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4 数椐库 t 珣器 


本 




事实还是推论? 


命 



你作出的每个决定都会有其原因。 在 Rails 中,知道如何作 出明智的决定 

可以节 W 你的时 M 和褚力。我们会/£本«来召一 F 用户的需求坫如 
的一开始就影响你的选择的 # 你应该使用支架然后洱修改生成的代码？应 
该直接从头开始创迮代码?无论哪种方法，一旦你幵始定制自己的应用, 
你就必须 r 解查询器 (finder) :通过一种有 .£ 义的心•式宋•获取数据沣由此 
满足用户的需求。 
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用户需求 


Ruby ville 健身俱乐部让你保特体形 


Rubyville 健身俱乐部 Q 豪子有能力为每个人找到最佳的课程，最近 
他们启动了 •项私人教练服务。这项服务的需求 a 很大……大到教 
练 ff 】 无法记录下他们所有的客户。教练们需要你为他们搭迮一个应 
用，并且尽快。 



私人教练们需要•个 Web 应用来让他们快速且方便地管理毎个荠户 
g 旮课程。•开始这个应用要能够列出每个客户的健分课程信怠 
许客户添加、史改和_除记录。 F 面是主页面的 草图： 


[nfSo 





\，Uf -//loUl^ti ^OOO/cUni MjrkoMis/^i^a/ 


i 

Lis'tin^ elieivl workou*ts -for Lenny 

Tramcr Pi»ra4i**i _ha •( wnfc •七 Paid dMMri 的一名 1 户 - 

Omi 10 

100^^01^ UTC V9J0 



Brad 10 

100^-07-11 (A ^OO UTC 25*0 

SKgw £d»i Daiwy 


S 咖 

iooi-oc-oz (y)nx)o utc 艿乃 



AlarsKjll 1^ 

looi-oi-zi m^oo utc \^o 



Omi 10 

100^-10-01 CAM CO UTC 1^0 

SKow^liiDjsiinBy 

10 

1001 - 10 - 0 ^ nxx>x>o UTC v>o 

Skgw^vjpciir^f 


Ne^n dtiewi_>i«ortoni 



- , 
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数据庳査询器 



先让我们从使用支架为下面的模型生成一系列页面来幵始这个 应用: 

client_workouts 



我们应该使用怎样的 scaffold 命令呢？请在下面虚线处写出答案 9 然 
后运行这个 命令： 


在 ci/b 荽下伪的 





现在，再看一下教练们希望这个应用实现的功能，然后写下该应用 m 要实现的功能与你 
刚创建的支架式 ( scaffolded ) 应用版本实际功能之间的 差异： 


你现在的位簠> 
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该不话.使用支架 



让我们从使用支架为下面的模型生成一系列页面来开始这个 应用： 


施 


呓 fi 长 f S 广 iiiH 汰个 
frfi Trails 6 ^ . 


client workouts 


client name 


duration 一 mins 
date_of_workout 
paid_amount 


类型 

string 

string 

integer 


我们应该使用怎样的 saiftbld 命令呢？请在下面虚线处写出答案。然 
':)， 后运行这个 命令： 

% 二硌 i 宅钕 .fstflffoLd cUci ^ t _ worleow.t clienttra ' u ^ tr - stni ^ 

冲条表爸机 0 * ciuratio^ y^iy^dv^tc^cr date of wo\rVu>ut：datt paid avu，ou^：dtci^vMi 

- wtee ^ to ：^0 rRlc 


你的任务是找出应用需要实现的功能与支架式应用实际功能之间的差异。 

.f.*/P. *.t 


支架式应用 # 不正碥 —— fs 是我们应泫 t 行编 
写代码还是修改支架代码？ 

支架式应用不符合我们的需求。我们前闹肴过,手工创违简单应用 
£容易.根本不需要支架。不过 S 种做法是先创逮 t 支架式应用， 

然后对 Rails 生成的代码进 h 修改或荇添加。 

那么我们应该选择哪种做法呢? 

劈一 

如果你还没有使用支架来创建健身应用,那么我们可以把这个应用与 
教练们所需要的应用比较一下，看看我们有多接近。 
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数据库查询器 


迖个应用其实 f 起來很狻近教练们的 f 求 

牛.成的代码中有-部分石起来与教练 fn 恕耍的页向很相似。索引 M 
舶列出 r •组与教练们要求的箪本•致的 数椐： 


达 4 鈦遙们矣嘍 
/ 卷的的 …… 


nno 

HDS 回 S 


CIkou-fcs • findT " 


|» 七知 :/ AetiWwsi^OOO/ dlien^__workovts/ "find/ 


Lis^ivt^ dlievrt workou-ts -for Lenny ^oldbetr^ 


Tra'mer D^ra'iion ‘ms Dd'Lej 

Ctmi 10 lOO^Ol] 

Brad ZO 
Svevt ^0 
^lirsKall I 弓 

Clmt 10 
Sara 10 


100^01 

ZOOUH 

100^10- 

ZOOI - IO - 


Uct * d*cw£ wcrkoM-fc 


•••_._ … 
用的 f A 否 


n o o 
(r 卜 j 


R ] 


ClientWorkouts index 

//local host:3000/dient_workouts 


Listing client—workouts 


Client name Trainer Duration mins Date of workout Paid amount 


Kirk Stigwood Clint 

60 

2009-10-05 

50.0 

Show Edit 

Lenny Goldberg Clint 

30 

2009-07-14 

25.0 

Show Edit 

Lenny Goldberg Brad 

30 

2009-07-19 

25.0 

Show Edit 

Lenny Goldberg Svert 

90 

2009-08-02 

75.0 

Show Edit 

Lenny Goldberg Marshal 15 

2009 09 29 

15.0 

Show Edit 

Lenny Goldberg Clint 

30 

2009 10-01 

25.0 

Show Edit 

Lenny Goldberg Sara 

30 

2009 10-05 

25.0 

Show Edit 


New client workout 


不仅生成的 IS 面看起来像我 ff 1 需要的页面，而且我们知 
逬 i 架式应用为我 们实观 r 对客户健穿 i* 程数据的常 m 
操作。换句话说， i 架式佐 用默认 允作我 们创 im . 读取. 


史新 和刪除 M 录。 


職 • 在这种情况 F， 是修改支架式应用坪加入我们需 
要的修改奸，还是从头开始做，就像我灯为 MeBay 做的 
那样更好? 












当我们创速 MeBay 佐用时，我们决定不使用支架。这是因为客户要 
求的内容非常简单，比较起来，从头创建应用史衮易。他们要求的 
功能比支架提供的要少得多。 

何这次我们需要使用所有的 CRUD 操作， iMRAUN 还要搜索 单个客 
户的所心健身课程安徘。因为我扪需要 史多的 功能，使用支架作为 
砹用基础就可以帮助我们实现人部分功能，然 G 我们可以在生成的 
代码 HI 加人更多的内容。 


这泉 Trails 的? I … 

佟祛 g 历的的间爹•残 
们軚殆夸&發 
劝钛 ~ ) 


耶么我们该如 M 杏找并显示单个艿户的 fit 牙课枴划作？ 





. 杏询器 


设 i 十揸 索功能 


搜尜功能 n 像 f 面 这样: 



O 控制器 k 的 find 动作，该动作完成实际的搜索。 


O 新 的结* «面，带有指向每条客户健讣课程信息的链接 

列表-这有点像支架生成的索引页面。 


那么我们应该从哪儿着手呢‘? 


你现在的位置 ► 
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从表单做起 


让我们从 建交表 单孖始 

我们需要创违一些新的组件，所以让我们先从用户界面开 
始。这样我们就能够从教练那儿获得一些早期的反馈。下 
面是教练们将用于査找客户的搜索表申 .： 



你之前已经建立了一些带有表单的页面。你能够看出来这个 
表单与其他表单有什么差异吗？ 

看一下我们刚刚为应用生成的其他表单：创建和编辑表单。 
它们有着与 ClientWorkout 模型对象的域 ( field ) 匹配的域。 
这次我们没有能够匹 gd 搜索表单的模型对象。那么在没有模 
切做基础的情况下我们该如 M 创建表单呢？ 


搜柰需要一种新表单 

我们不希望通过校型对象来创违表单，但是我们使用的 
form _ for 辅助函数需要-个模型对象。我们该怎么办？ 

幸运的是还存在另一个辅助函数标签，它能够创建无需模 
型的表单 这正是我们当下想要的。 
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数据库査询器 


搜索砉单代碚冰箱糍锈 

你需要创建一个搜索页面模板 a 问题是 在冰箱门上刚刚完 
成了代码的主要部分时，有人砰地关门导致所有的代码冰箱磁 
铁都掉下去了！ 

值得庆幸的是模型无关表单的代码与樸型相关表单基本一致。 
你能弄清楚代码应该长什么样吗？ 


<% %> 



%> 


<% . %> 


•/client workouts/find" 


submi 


ubmit.__tag 


GD 




— forip^tag^j 


GD 


，，： 


search_string 



你现在的位置 ► 
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没有〔 , H . 表单 



搜索皋单代碚冰葙糍铁解箸 

你的任务是拿起代码冰箱磁铁把它们组织成创建搜索表单的代 
码，而且不需要任何相应的模型。 

你能弄漓楚代码应该长什么样吗？ 


我们坏錡射 Jl 的珀作 W 


,<% 




form—tag 


'/client workouts/find" 




•任 4 哒 11 


text—fie Id 一 tag 1.1 ■ search string t =•；> 


»• <%= 


submit tag .. 


"Search" 


%> 


end 


^v). - t«e H'l 


© ^ T - f f 


i 1 ®)- 我不明白为什么我们不能直 
接使用 Rails 的支架它不是应该生 
成所有的代码吗？ 

支架 （ scaffokling ) 只能提供 
基本的钊建 （ Create ) 、 读取 （RtriKO . 
史新 （ Update ) 和刪除 （ Delete ) 操 
作,，大多数应用需要的不忟仅是 
CRUD 功能 

为什么我们在 MeBay 应用里 
没有使用支架？ 

我们没有在 Mehay 里使用支架 
是因为那时候我们只需要一个只读应 
用。对于非常竭单的应用来说.手工 
刨建汴的应用会更容易也更有效率 



苕蠢碰 


I 1 ®)*我们可以还是让支架生成应 
用.但是删除我们不需要的功 能吗？ 

是的.当战可以这么做。你可 
能都会使用支架开始大多啟应用只 
有那呰你需要很少功能或者功能与支 
架式代码有很大差异的情况下.你寸 
需要手工釗建 应用， 

1^1 • 所以有时候从头开始更好. 
而其他时候则最好修改支架生成的代 
码。那么我在什么时候使用哪种方法 
呢？ 我该如何确定哪一种方法最好？ 

如果你将要使 fflCRUD : 刨 
達.读舣、更新和刪除的大部分操作 
时，清使用1架.你可以自行 判断： 
哪种更快一痊手玉编写代码还是 


刪除那些无用的支架生成代码” 

I 1 ®)"什么是槙型相关表单 （model 
form ) ? 

模型相关表单是 指表单綁定在 
幞絮对象上,这种表单被显示时.域 
值将来自于模塑对象的美性， 

而樓型无关表单 （ non-model 
form ) 是什么？ 

这是没有蜱定到模型对象的 
表单。糢型无关表单被 用于一 组特沬 
的域值.这些域值通常 甩于如 搜索表 
单或者其他不需要保存到数据库中的 
啟据 
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查询器 


为界 i 渌加搜 f 功能 


现在我们 Li 经釘了搜索表单的代码，我们砲该能够为 它创迖 -个仝 



捜索通常是一个功能.而不是一个独立页面。 

人多数 n 站会打 个 内裨在毎！»电的搜尜功能，所以我 扪也 " r 以 
这么做。 a 们 " r 以把拽索添加到毎〖;(的右卜.方，然尉保持这一页的 
tt - 他内奔不变^ 


把代码添加到毎-页意味着会有很多重复的代码需要维护，但是如 
H 足把新的搜尜 ft 码添加进 •个文 汴呢？ 



你现在的位 fi > 
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布局是共享的 


#笔上阵 - 

解答 我们可以在某个文件中添加搜索表单，使得它出现在应用的每 

个页面上》这个文件的名字是什么？请在下面虚线处写出你的 
你 2 记饵年 爸唾， 它们 答案。 

只耷莕个否面梭板邾含 

忽含的衫 . A flpp/v/Uws/U y out^/ciUt^t_wor\fZouts. ktwtl. crb 


• - ^ 6 

这是 你雜储 S 

app/views/layouts/client_workouts. html .erb 布局文件里的代码。 

请从 www.headfirstlabs.com/books/hf rails 下载该文件 




<! DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 

"http://www.w3.org/TR/xhtmll/DTD/xhtmll-transitional.dtd M > 


<html xmlns=’_http: "www.w3.org/1999/xhtml" xml : lang»"en" lang="en"> 
<head> 

<meta http-equiv= H content-type" content^"text/html;charset=UTF-8" /> 
<titXe>ClientWorkouts : <%=■ controller .action name %></title> 


<% - stylesheet_link_tag * scaffold* %> 
</head> 

<body> 



冻扣一个样趴象 


<span style=" text-align : right■’> 

<% form—tag M /client_worJcouts/find" do %> 
<%= text 一 field 一 tag : search 一 string %> 

<%= submit_tag "Search" %> 


<% end %> 
</span> 



㈣ || 料 _ 




<p style= H color: green"><%= flash[ : notice] %></p> 


<%= yield %> 


</body> 

</html> 
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查询器 



修改你的应用的布局来让它包妗搜索功能。 ’*1 你刷新应用的每一个 K 面时 
会显示在右上处。 


搜索域应该 




兴 C 射钱亡扣栌$旁一 
域(象 Ci 樽 S 孑在荽个资 


但是我们该如何仿问拽索域的内容呢？下面是在每个页面中巾•成的 
用 i : 产生搜索衮中的 HTML : 


既然你知道表单对应的 HTML 是什么样子，你认为你可以在 
控制器代码中使用怎样的表达式来访问搜索域的内容？请在 
下面虚线处写出你的 答案： 


你现在的位置 ► 
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使用参数 



那么表簞参数是否有不间的结构喂？ 

我们 在前 •琪里 tt 用的 form_for 轴助闲数创达了一个模型相关表 
单 也就是一个基 r 模! af 象®性的 HTML 表单^当校型相关表 
黾彼提交时， Rails 知道你可能嬰把域浪转换到模型对象中去。例如， 

当支 架式& ^表电 （由 form _ for 创达 而成） 被提交时，它会像 T 
面这样组织 参数： 呤容 



workout 】 这样简笮的表达式来获取域值的哈希结构。 一 


fHform _ tag 辅助函数乂是做什么用的呢？ form _ tag 创违一个模 
型无关表单 # 这是一种用来编辑一组特殊域值的表申.。 因此， 搜索表 
单（由 form _ tag 创建而成)创违如下结构的请求 参数： 托锖求参 
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-V ’;. 据库查询器 



现在你 Li 经知道如 H 从表争中检索拽索字符串，是时候在柁制器中创迮服务器端代 
码来实现搜索。 酋先我 们会把搜索域中的值敁示&控制台上。这样我们就能验证表 
中.是否正常 r 怍。 

这个表中.将被提交到名为 /client _ workouts / find 的路径。在 config/routes 
rb 屮的默认路由将能够 ' MU ]映射这条路山。请_出卜而唧条默认路由会陂 使用： 


map.connect • : controller/ : action/ : id* 

map.connect • : controller/:action/ : id.:format 


接 F 来,在控制器中创逮•个 find 方法来显示搜索域的内容。 提示： 你可以使用 T 面 
这条命令在柁制台窗 I 丨 显示一个字符串。 
puts <string> 

^ ii 衾蓍 web 磉务器的穿0中 
^ •子 穹符寒 


你现在的位置 ► 
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使用 puts 



现在你已经知道如何从表单中检索搜索字符串，是时候在控制器中创建服务器端代码来 
实现搜索。首先我们会把搜索域中的值显示在控制台上。这样我们就能验证表单是否正 
常工作。 


这个表单将被提交到名为 /client _ workouts / find 的路径。在 config / routes . 
rb 中的默认路由将能够为我们映射^条路由。请画出下面哪条默认路由会被 使用： 


我 fy 不 f 龙钇$ 

戏 ft 用遠条软认 
31由 


-^ 


map.connect 
map.connect 


* : controller/taction/ : id' 

' : controller/ : action/ : id.:format 


接下来，在控制器中创建一个 find 方法来显示捜索域的内容。 提示： 你可以使用下面这 
条命令在控制台窗口显示一个字符串。 
puts <string> 

H 赛 web % 务器的 穿。" 孑李锊 

$ 

我科类炎的軚差写出正 

puts ’一一 
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:，心 查询器 



U: 我们来测 W 下这个代码。在应用的任意•豇输人一呰搜索文本, 
然后点 A “Search ” 按钮。 



Listing client 一 workouts 


H 


这基 4 你轄下搜素接茌后 
/ 主的辜鴂。 

\L 



请不要袒心你在按下搜索按钮 cf； 到的错误。 这仅仅坫闪 为我们还 
没有创建一个搜索纟, V* 页面校板。值得关注的输出会在你运彳 iWeb 
服务器的那个控制台上,在一些缺少结果模板的错误中，你可以看 
到如 K 的 内容： 


冼 ㈣' 
-^ 



Rendering client 一 workouts/show 

Con^>leted in 9ms" - (View : 4, DB: 0) j 200 OK [http : //localhost/ 
client 一 workouts/1] 


Lenny Goldberg 


Processing ClientWorkoutsController#find (for 127.0.0.1 at 
2008-10-13 22:00:40) [POST] 

ActionView : : MissingTeinplate (Missing ten^>late client 一 workouts/ 


所以我们已经在毎个页窗上创建丫搜索表单，而且 我们在 服务器 
h 也有-些代码能够读取用户正在搜索的字符串。 

现在我们需要真正地实现搜索 
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回顾 


复习要点 - 

■ 应用经常需要实现比创建.读取、更 ■ 你在 form _ tag 表单中需要使用__ tag 
新和删除记录更多的功能。 域。 

■ 有时你需要设计自己的页面序列，最 ■ params [: field _ name 】 将给你模型无 
简单的方法就是首先考虑用户的想法 关表单中 名为 ： field _ name 的域的 

以及他们将如何使用你的应用。 值， 

■ 如果你需要一个并不匹配任何模型对 ■ puts "A string " 会把一个字符串输 
象的表单，那么你需要使用 form _ 出控制台。 

tag 而不是 form _ for 。 


-'雜蠢网 

卢没有默 


- 


我什么时候应该使用 form _ tag 而不是 form _ 


我能读取 form _ for 模型相关表单里的单个域 


如策你的表单被用来编辑没有存储在模塑对象 
中的啟据时.你需要泛用 f6rm _ tag 。 我们对搜索表单 
使用 form tag 是因为没有哪个樓空对象带有单一的 

search _ string 属性。 

^ • 为什么 form_for 和 form_tag 辅助函数要使用<% 
... %>而不是<%= ... %>围起来？ 

form 辅助函啟用在脚本段(<%...%>)而不是表 
达式(<% = ...%>)里是因为它们不仅仅是 iSj 单地生成 
HTML 。 还记得我们为 for 饨环使用脚本段吗？这是因 
为 for 摘环控制撺环体里的代码内容。同徉，表单“控 
制”它们内在的域辅助函軚的 HTML 的生成. 

这听起来有点复杂…… 

^ • 没关系 你并不需要现在就理解它。你只要 
记住通过脚本段来使用 form _ for 和 form _ tag 也 
就足够了 。 


• 盎然 client _ workout 炎单中的域 
值可以通过 params 【： client _ workout : 检 

索到 而该据只是另一种哈希而已 u 所以 


如果你想得到 “tr 

params (: cl ient 


iner " 域的值.你只需要使用 
workout ] {: trainer 】。 


对于 - find " 动作，我们使用了一条默认路由， 
我们前面为什么不这么做？ 

不行我们前面使用的路由不足以匹 Si 默认路 
由。 

1^)* 我如何知道何时该创建一个自定义路 由呢？ 

• 如果你在命令行上输入 rake routes , 你会看 

到应用里所有可用的路由。如果没有一条路由能够获 
得匹配.或者存在一 条匹配 了错误动作的路由.那么 
你就需要添加一条自定义路由 
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查询器 


我们如 何查找 窖户沱录哝？ 

我们读取特定客户的记录时会有什么 N 越吗？到 B 前为止.我 
们只读取过返回的单条记录，或者数据表里的所有 id 录。这次 
有什么不冏吗_? 


o 锿取单彔纪录 


o 锿取所冇记录 



如果我们不传入 id 号而是 
传入特殊符号 rail , 那 
么模型将返回包含底层 
数据表中所有记录的一个 
数组。 


这次，我们要读取匹配特定搜索条件的记录。我们 


我们能够通过 id 字段 
中的值来读取单条记 
录。我们知道通过这 
种方式返回的记录只 
有一条，因为 id 号对 
于每条记录来说都是 
唯一的。 


O 锿取 E 紀特定彔件的论录 


latitaiiw 




可能希望返回的不是一条记录，而是 一个模 型对象 
数组。但是我们不希望得到每条记录的模型对象 
我们仅仅需要那些能够匹配搜索条件的记录。 


—•参 。切 姑 - 

想一下底层数据库表中的数据。能够匹配捜索条件的数据表中的记录是什么意思呢？有 
没有什么方法对匹配记录返回 lure 而对其他记录返回 false ? 


你现在的位置 ► 
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大海捞针 

我们 R f 要那些 cliewt _ wa » ne s 
搜索字符$的记录 

教练们想要搜索某个特定客 广的所 打饨身课程倍息。模切需要一个简 
黾的能够时队配记录返回 truerfii 对其他 i 己录返回 false 的测忒闲数。比 
如： 


Is ClientWorkout.client_name = params [: search string] ? 

v -- —-— - J 

#入 的内容 u 


如果模型紂数据表中的每条 ill 涂邰执？？这个测试函数，它就能够找 
到数据表中所有的匹配 记录: 



一般来说， 我 们需要 - 个查 洵器， 它能 够找到数据丧中拥冇特定字 
段中的特定 fft 的所存记录 D 
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紱据库 査询器 


毎 f 厲性都布一个 t 沟器 

大多数应用都需要查找在某个数据库字段中有特定值的所釘 记录， 

所以 Rails 把这一 切简笮化了。 

但是怎么做呢？校型代码针对自己的每个 域性都 冇一个奄洵器。你 

不需要亲手添加这些査询器—— Rails 提供广。所以 CliemWorkout 模 达签及科的樣每 H 



f 洵器古迗一个乐右2髡时 
象的 數迷 


还 U 1 得模型对象中的-个属性会被映射到底 y 数据表的-个字段 t 
吗？所以毎个®询器都可以用来査找所有拥介特定字段中的特定值 
的记录， 


.磨笔上晬 




请完成下面 find 方法的代码。 


def find 


@client_workouts = 


end 


你现在的位 S > 
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find_all_by_attribute_name 



接 7 采哝？ 

现在我们巳经有代码来找出匹妃搜尜条件的所有 id 录，所以我们需 
要把这哼结*:显示给用户。该怎么做呢？ 

我们需要创建一个 find.html.erb 页面来显示搜索结果。 


足 ClientWorkout.client name = 
params [: search_string]H , i? 
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玷据库査询器 




你现在的位置 ► 
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显不+客户— 




为 fin _ d 方法创建 一_ 个页面模板籴显示如 T 的健身课程信息列表 


赚 


<Trainer name > < Wo - rkaut ' durat " ion > <Date of the workout > < Amount ' pcuid> 


提示： 支架已经生成的索引页面与你需■要产生的页面很相似。 

— 也； 4 * 系 


.<hl>filtci>vt _ wcrteowits for ra wist ： sea rch_steiy^J ^></h±> 


< th > TraL ^ r < Ah > - 


jCth >t>w.rfltLOKV kvcli^s </tli >. - . 

,；;;. 乂 .... r .... ............ r . 

<th >T>att[of workout<AV\ > : . S 二 


二 m •… 二 ;.：.“'… 




<tV\>Vald av^.ouh^A:</th> 


























现在搜索功能佐泫 " J ■以工作了。让我们试 R 它 


数据库查询器 



你现在的位 S ► 


M http :// k ) calhost :3000/ client _ workout ! - Q - 


Lenny Co dbeig 


这会如何改变搜索条件呢？ 


50 2009-10-05 50.0 Show Ed»t Destroy 

Edit Destroy 


ClientWorkouts index 


ant workouts 


Cl^nlWo^kOwtt find 

< - C '<■ ■*■ Mfino/ftotiiiioii 1000 tK>n.iw»ikogij^'»a 


Listing client 一 workouts for Lenny Goldberg 

Tr»lMr Duration mtns Oat« of worttout Paid amount 

Olrtt 30 2<X»-07 14 25U> Show gd«t P»«ttov 

brma SO 2009-07 19 2S,0 S&fiB Utl QSSUfiX 

Sv*n 90 2009-0B-0} 7S.0 b&t OSJSlStt 

n»»n«b IS 2009-09 29 ： S.O Show Edit Dertrov 

Oinl 30 2009-10-01 2S.0 Show Edit Oast/ov 

Sara M 2009 10-OS 23.0 Show tail Ontrof 


Duration Date of Paid 

mins workout amount 





匹配这个或者那个 

我们 t 要四紀窖户名字或 f 敎练名字 

使叫特定客户名字杳找 id 录的搜索功能很成功。 m 是如裝搜索还要 
根据名字杏找教练，那■么它对海条记录运用的测试逻辑会史 加复杂 
一些。 

我们不 再使币 


client 一 name = pa rains [: search 一 string] 


现在 志要的条件应 该是: 



iSfOiit t | 找鈦 豸的记采 . 



戍在 an 含洱利害户軚 
终 ..... … fsiil 如何 xnt ^ ： 

你看出迖几的问超5蚂？ 


模型对象中的毎个厲性都有一个查询器。每个査询器都打一个运用 
至数据库中记录的简单测 ut , 根据给定的值來检査数椐库屮的单.个 
字段。但是现在这个测忒更加复杂了，有没冇什么方法来指定丧洵 
器需要运用至数据库中 id 录的测试呢？ 
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数据厍査询器 


t 沟器生成数椐库 t 询 






那么査询器是如何 r 作的？，你运行某个许询器时会发生什么？ 

资洵器的 r 作就是袢你跟数据库进行交谈。记忭:，当査洵器被调 
用时，它会使用•种名力 SQL (Structured Query Language . 结构化 
査询语宫）的 iSa 来生 成针对数据库的资询。 ^_____ 

《.:S 入 _Adis<2<4 1 



ClientWorkout.find_all_by_client_name ( 1 Lenny Goldberg' 


SC^JLfif) 


SELECT * 


FROM client wo,xJtQufe- 

- - - 

WHEREvclient name= 

'Lenny Goldberg *) 


- —— 






灸询被数据库用来丧找数据表中所有匹配的行。模兜把这呰行转化 
成模型对象并通过数组返回给控制器。 


根据査询器的工作原理，如果我们要搜索客户或者教练，需要修改 
哪些地方呢？ 


你现在的位 M ► 
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控制 SQL 

我们 f 要修改 SftLt 询中的条件 

我们需要-些方法来告知模型生成如下的 SQL 査洵： 



但是 SQL 杳询中的条件是由查询器方法来生成的。 ftff ] 町以把字符串 
传递给查询器（比如 “Lenny Goldberg” > ， fO . 我还没#修改过发送 
给数据库的 SQL 条件的实际结构。 

可以修改 SQL 査询参数会足一件很重要的事悄吗？好吧一是的，它会 
足的。那些査找特定槭性中的匹屺值的许洵器很有用，但是指定 SQL 
条件能够 U ： 你做 电多的 半情。 它可以让你湲 盖杏询器的默认行为，完 
全控制模型需要访问的数据。这也正是我们 ft 这几所霜 要的： 对 SQL 
丧询的更多控制。 

那么我们该如何修改这些条件呢？ 
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数据蒗查询器 


健用 : conditions 來提供 SQL 

为每个属性生成的査询器作常简单易用,但是它们的问題在千不够 
灵活。你经常盂要对数据库进彳? 电复 杂的杳沏。 

因此，所办的丧淘器都允 iT •你传人一个名为: conditions 的命名参 
数，它可以包 念被添 加到 ffi 洵器所生成的 SQL 中的额外条汁。 

r 面是能够实现教纷：/荠户搜索的•种方法： 


@ client 一 workouts = ClientWorkout.find( : all , 

: conditions=>["client 一 name = 'Lenny Goldberg' OR trainer = 'Lenny Goldberg'"]) 

— 

条 4 枣教 ，攻 设 I 


此版本的仵询器将返 N 所有教练或者客户名为 M Lenny Goldberg” 的记 
录，不过 你能行 出这儿冇什么 H 题叫？如*:我们想要搜索其他人 ifilH 
Lenny 呢?我们真正想捜索的是记录在 params [: search _ string] 中 
的人名，该怎么做呢？ 


幸运的是， Rails 正好有一种可以灾现它的方法。这种方法允许你像卜面 
这样把条件参 数化： 


@client 一 workouts = ClientWorkout. find( : all, 

: conditions=>["client 一 name = ? OR trainer = ?''/ ，色兮族蟑 
params[ : search_stringj, params [: search—string]]) 




A 


条件数组 Jr . 第一个字符串中的“？”会依次被锌换为后续的值。这 
就怠味笤査询器现在能够为任 W 搜索参数屮成 iH 确的 SQL 语句。 
无论搜索哪个教练郎可以返 M 相砬的 id 录。 




它的效果如何呢？ 


你现在的位釐 ► 
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请找 








査询器 


—— + — 

健讣俱乐部开始记录所有在他们的户外棒球场进行的体脊运动。请 
把数据库奋询器 与 它们相应的用途连接起来。 

查询器 用途 

BaseballGame . find( : all, 在淡乎进彳 j ■的体育^动 

: conditions=>[ 

'month no > ? and month no < ?', 

9, 3]厂 


BaseballGame. find (: all, 啦实 h . 这个 洵不 会 返回任 H 结 

: conditions=>[ 1 ft , 所以它不会被使用 

'month no > ? or month no < ? *, 

3, 9]) 一 


BaseballGame . find( : all, fF.flFUtt 行的•运动 

: conditions=>[ 

'month no > ? and month no < ? *. 

3, 9] 厂 


BaseballGame. find (: all, 这个似旬只返 l u l 所打的 I 己录，所以 

: condi tions=> [ 它也不会被使用 

’ month no > ? or month no < ?', 

9, 3】厂 


你现在的位置 ► 
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了解你的查询器 


— + + 连4著 

你的任务足把迕洵器与它的用途连接 起來。 


+ _ 

解奢 


查询器 用途 

BaseballGame . find( : all, 在淡乎进行的体 ff 运动 

: conditions=>[ 
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数据库 査询器 


然后就有一个人在…… 

正当你演示锒个系统时，有人在敲门。这 个人来11 于陡分 
供乐部。 



O 0 




ft ， 说鉑们类®球痛枝 
(r£>icl r«0c) -聶 g 帑 

玢够 C 藉••卢间&…… ' 


看来输入到系统中的数据发生了问题……请为下一章做好准备. 
在那儿我们会深入了解这个健身问题。 


你现在的位置> 
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rails 工具箱 


Ij^t' 


你的 Rails 工爯箱中的工異 


? 你己经把第4章收入囊中了，现在你已经将选 
择是否使用支架以及如何聪明地为你的应用选 
择正确的数据的能力加入了你的工具箱。 


• r ^ ILs X U 

的 SOJU 0 

主威无乘绑玄到 ㈣ 的象 
的阑 i 表輩。 

^ w-by J |2 


pKts <Stl^0> 杏控制台 ((*^4 
呢务器的 邵个）上 $ 泽一个 玄 
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5 验证你 的数提 



每个人都会犯错……但是很多错误都可以被避免！ 即使你的 W 户令神资注 ，他 

们依然 "了能把错误的数椐输人到你的 Web 应用中，使得你不得不妥善处理它们 。 m 
足 w 设想一下，如*:有一种方法 "r 以在第 •时叫 阻止错误的发生会如何呢？这就是 
为什么我们要引人验证器 (validator) 。 请继续读 f 去，我们将为你展示如何添加 
聪明的 Rails 验证器到你的 Web 应用中， 使你能 够控制 f| •么样的数 W •被允 IT •输入以及 
fl* 么样的数据应该被排除在外。 


防止错误 




亲爱的，如果我们袞 
绅所布的銥姆礼粕，我们 
饫能兵浔起我一£箩想拥 
荀的木蛭农坫《> | 
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错误数据 


注 t — 应用里布铐误数椐 

私人教练的 Web 应用看起来一切正常， ft 到健身者的 出现。 健身 
者说他们 d 经支付 f 健牙的 费用，并且有收 据来证 明这点 ，但 
是他们的付款汴没打在系统中 出现。 



什么地方出错了呢？ 

点击保存按钮后应该把数据保存到数据库中，但是有地方出错 
To 系统没有把支付金额保存为$50, 相反, 它保存成了01 
为什么会这样？ U : 我们看粮个事件的发生过程吧。 



一名教练输入“$50” 到视图 
的支付域，然后点击保存按钮。 
值 “$5 cr 被传给了控制器。 


O 控制器接收到“$50” 的支付 
金额，并把它传递给模型。 


0模型接收到值“$50”，但是有个 
问题。由于这个值包含了一个$符 
号，模型无法把它转化为数字。它 
反而把 0.0 保存到了数据库。 



这个问题足由十教 练在 ㈣ mvh 输人 r 错误的数据，而 aw 需要防 ih 这 
种错误洱次发屮。我们需要编 s 代码从而 m 表电数据被入数据库之 

前来验证它 但验证器的代码应该放在哪儿呢？ 
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验证你的数据 





Mark : 验证器放作哪) L 有 ff 么关系呢？ 

Bob : ■我伯需要在错误发生的地方修正它。错误发 生仵表 单:里，-所 
以犮 中就 足它^泫被修 il : . 的地方。 

Mark : t > ?L rt : 接把柃 作放 ^控制器 甩。 控制器丨_说存和 Of 
跅"达。 ftffM 以 U •.柁制器柯能力决定足？ f 保存对象~ 

Bob : 但是问独发生在表单里。 

Laura : 我不确定。 H 题真的 ft 模蘩里吗? 

Mark : 梭型 H 足数据而巳。我们编写的大多数代扔都在控制器里。 

Laura : 我认为模型不仅仅是数据。难逬我们不能把代码也放在那 
儿吗? 

Mark : m 足逻辑部分郎 作控 制器里。 

Bob ： > f ； 可以史 M 巾. •些。 ft ; 只志要把•个 JavaScript 检仵放在 




疚耜： 


表蟓的标 i 己里_ 

Mark : 如采用户把浏览器甩的 JavaScript 关掉呢? 


Bob : 嗯，轻 fe 点， 没打人 会关掉 JavaScript 。 


Laura : fll . 是你怎能依赖 F 这•点呢？特別足代码放在哪儿是很明 
敁的时候。 


Mark : 控制器。 


Laura : 模1*!。 



你现在的位置 ► 
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在模型中进行验证 

M 笔上阵 - 

解答 你认为验证应该放在哪儿呢？请在下面虚线处写出你的答案 ^ 

学畔料 K 中辦郎:绔熳可! if . 芩今埤殚數*粵♦璆® 巧 i £ ^. 9 ft • • 


验证代 码故在糢型中 


把验证代码放 到视阁 或者控制器中的问题在 r •两个不 mi 地力的 代码郎 
会试图将数据保存到数据库中》设想-下，如果我们在拕制器中有插 
入和编辑方法,它们都需要验证器。如果验证被放在模型中,那么就 
无所谓数据如何保存了 对该数据的验证始终会发生。 


插入错误的数椐 


^ 私器 1 ：的钵入 
{%% . 


— . 


毛 > 5秸入表辈 




Create 



new.html.erb 


编緝错误的数梅 


s 餘入 •？ 箝谓的 
敎轉 


Paid amount 
丨 S 50| 

• ’Create' 

Back 



乇命嫌一个硿 軔器技 作仗 




edit.html.erb 


通常来说 T , 在模型中进行验证适较好的做法。屮盘，这也足我阁为什 
么会有模型层的•个原因。 模型不 W 仅是数据。我们把数据库封.装到这 
一代码层的原因在丁我们 ^以添 加一些数据库本穿不 ft 备的吏高明的手 
段就像验证器那样。我们该如何把验证器添加到模型屮呢？ 
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Rails 使用验证器来实现简单验证 


验证—的数据 


鉍个 系统郞需嬰对输人的数椐执行一呰检杏，冇时检许代码 “ T 能会很 
长很复杂。 

那么 Rails 会如何帮助我们呢?毕竞,你需要做的检査都相当个性化, 
不是吗？好吧 足 HH 〈坫。针对你的数据的骀 ill ： 规则 "f 能足你的系 
统所特有的。但是毎个规则可能检査的是-些典型的错误，比如缺少 
数据，或奸错误格式的数据，或疗错误类型的数倨。 

这就 ii 为什么 Rails 提供 T •组名为验证器的内嵌的准检查。验证器 
是-个 Ruby 对象，该对象奔看人们输人的数据并对该数据执行简单的 
检査。它什么时候执行检赍呢?每当存人试图保存或者更新数据痄中 
的数据时魷会发生。 



验证器是改进你的数倨质茕的快速且有效的方法。它们会帑助你过滤 
数椐，确定哪些可以而哪些又不能进人你的数椐库。在数据 W 错 i 乂的 
时候，它们甚至町以提供-纟 IL 错误消息来帮助用户诊断错误原闪。 


但验证器是如何工作的呢？ 
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放大验证器 


验证器 I 如何工作的? 


lh 我 fH 跟随 CHent Workout ^ W 遍验 tt 过程。 



用户提交某个 ClientWorkout 详细信息 

问题 il：. p 与 id _ anvount 域 ti 含的足 “S50” ffti “50 
无法被匕成数卞值。 .. - 

Paid artiount _ ••: • - • . • " 





控制器把表单数据转化成 Client Workout 模型对象。 

该換兜村象存储 r 衣争数据的拷 w ， 而梭型徕 m 泫拷 w 来斗:成榷型的《性值《如果 
.你 iW « l 对象 0又:它的 paici _ a nK ) unt 慽性的 侦， 它将 试阌把 “ S 50” U 匕为•个数 
fft , 仇是它做心到，所以控制器就会说 paid _ amount 逄0.9。 


o 控制器试图保存该对象 

柁制器耍求校喂对象保存它 ft 己。通常对象会把 -3 UH 讣的 UU •保 存到数 
据库中，此时 paid arnountmmnmo ^ 但丛如*: ft 投切中灯验 ilF . 
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验证你的数据 


模型对象运行它的验证器。 

当模型对象被要求插入或者更新一条 hi 汆到数据茚中时，它 
注先运行它的验证器。验 i 止器会检査 CliemWorkma 对象内存 




模型对象决定它是否可以保存记录， 

只行 ft 所行的验证器都波运行过后校％对染才能决定它足否 " r 以将 id A •保存 
到数据库中。它会如 W 决定呢？它会许 S 是否有任何错误被创达 。 paid _ 
amount 验证器没饤通过，所以模切对象没行保 存记录 而坫 ft •诉拧制器出视 T 


错诶 


控制器让用户重新回到这个表单。 

控制器中的代码知道出现了错误，所以它让用户问到表单页面以便 ！"• 
错误能够被纠正。 
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数值 



iil 我们 t 金玲 


这将为每个 CHent Workout 村象都创 边- 个验证器的实例。 

验证器需要知道它即将检査的诚性的名字。就像 Ruby 中的其他名字一柞, 
厲性 ft 使 ill W 号 ( symbol ) 來纷出： : paid_amount 1 

还记得符号有点 像字符串吗? 符号总足以 - 个冒号（：)开始,而且通常被 
in r •指向凡他名字，就像域或荇厲性的名字。 

么切会如 M 在我丨门的应⑴中运作呢？假定控制器心-个名为 

@ c 1 i ent_wo rkou t 的 C licnt Workout 換 W] ■象。 

毎弋 控制器调 ⑴ @ c 1 i e n t _ w o r k o u t . s a v e 或荇 
@client_workout. update_a11r ibutes 时，模切对象将运行 

validates 一 numericality — of 验器<» 

如果模型对象里原来的表单数据将 paid __ amount 域的值记录为 “$50 ”， 
验证器将产生•个错误。 RaiLs 会发观这个错误并中数椐痄更新。 

这就是我们的理论。现在让我们看看这个理论是否能够奏效。 
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试髮 


器 Li 经就 绪. 让找们 测忒 • kuik 而。 .i、m 我们入 Ki:paid 

= v = ^ V; : v ； ^ ; :: ' 


( ft ，$50” 个足数卞，所以产牛.了 •条错误消息，这条健身倌息沧有被保存到数据库，布 
免户被返冋到 f 表取以修 iH 错误, 


那么 Edit 页面会如何呢?验证释能否冏 声发挥 作用?心:该能, I &为梭11! 作介 人尝试吏新数 
椐库小的 ici * 时会调⑴柑 M 的骀 III : 器。 • 
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验证必碧域 ^ . 

阁户在他们的健身表单中漏填 5 
一些数椐 


有畔人在表申.黾留下了一些空的域。例如，有个教练忘 f 在一些他 
输人 的健讣 课程电填写他 _ td 的名字。跅來，3他赍找所打 他负 责的 
雔身课稈时，.他就无法找到那呰没 苻填 号他名卞的 ill 垛。 


总这砹 的供:利的^ 羃. 
駟違 u(1Ferr 咩 
/不夺 ！1 •• 




fW 教练& f: 必袖•输入，•而 H. 客 A 名也偌要:输人。 Lenny Goldbep 就 
fr 几次训练课没仔输人他的 it:。Lenny. 通啬在./彳末收到账单， P 斤以 
当教练们 If 找 Lenny 参加过的还没有付款的训练课时，他们根4： M 法 
査询到。私人教练们无法接受这样的事情洱次发生！ 


那么验证器能解决迖个问题吗？ 

到 B 前为止，我们 gK 使用验证器来柃査输人值是否是数字。但是我 
们实阪上杆 捋整套的验证器， 它们能够做任何审，从检迕某个 值迠办 
出现在列表中到某个值 ft 衣屮的特定卞段中足斤难-。 
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验证你的数据 


我们淡如何检 t %领填写的域? 


有一个验证器是我们可以用来检查必需域 (mandatory field) 的值的。 

实现这一功能的验证器就是 validates_presence_of: 

validates__presence__of : field—naine 




这儿是带有验证器的 代码 : 




class ClientWorkout < ActiveRecord: : Base 
validates 一 numericality 一 of : paid_amount 

validates_presence 一 of : trainer 
validates_presence—of : client^ 

end 



的名穹部波成*。 




I 1 ®)- 如果某个验证失败.樓型对 
象还会运行其他验证器吗？ 

答： 问得好，答案是 ：会。 即使 
棰型对象知道在第一个失敗产生后它 
将能够中止保存操作.糢型依然会在 
告诉控制器任何错误之前运行其他的 
验证器。 

为什么要这样做呢？ 

想象一下你在某个表单上有多 
处锚误。通过运行所有的验证器，樸 
型对象可以确信你看到了所有的错误 
消息，这样你就能够在再次提交表单 
之前修改完全部的错误。如果你只能 
看到第一个错误，你就不得不多次提 
交表单。 

I 1 ®)-为什么模型对象要保存一份 
表单数据的拷贝？为什么它不直接使 
用常规属性来存储表单的值。 


答: 


■ ▲ _ i 河题 . 


樸型对象需要基于表单提交 
的原始字符串来运行验证器 .. 如果， 
比如说，它把 paid _ amount 保存到 
一个軚字爲性里，它将不得不把值转 
化为类似于 0.0 这徉的数据。这徉就会 
掩盖出错的亨实。 

但是当我询问_个 
ClientWorkout 对象的 @ client _ 
workout paid _ amount 值时，它不是 
返回了一个怡^的属性值吗？ 

答： @client _ workout 对象所 
做的是它会查 看士交 的表单域里的 
值 （ “$50” ） ，然后返回这个值的 
数字版本。每次你询问 @ c 1 ient _ 
workout.paid _ amount 时它都会 
这么做 

所以这就是为什么它 
把 “$50” 变成 0.0 保存到数据 库中？ 


答： 


型会根拢樓型对象的爲性值构造 
一个 SQL INSERT 或者 UPDATE 语 

句。 当它查看@<: 11611 亡一 workout . 
paid _ amount 的值时，这个值就是 
0.0，这也是它发给数据库 的值。 

1^)' 我可以让模型对象跳过验证吗？ 

可以。如果你调用@ 
client _ workout , save ( false ), 樸 
型将不运行验证器就保存对象, 


问： 


是的。如果没有验证器，馍 


我能够更改错误消息吗？ 

答： 可以——你可以把你自己的 
错误消息作为额 卟的 字符事来 提供: 

validates _ presence _ of 
: trainer , '' Where’s your name ?" 

错误消息是如何被显示的？ 

:在表单中有个名为 f .error _ 
messages 的标签 • 我们在后续内容 
中会看到史多关于它的信息。 
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ih 我们启动浏览器来试苕输人一些错误数据到 New * 电里。值得期 
待的是，我们的新骀证器代码会捕捉到所饤错误。 







验证你的数据 





对 r 这个系统找们 u 宙忠检作儿汴賀怡.但是 让我们承夷嘛 些可 
IH 的验 i 正器。 btk 着你惠开能够把 F 列验证器和它 ( N 的用途连接 起来： 


validates—length—of : fieldl , 
: maximum 二 >32 


柃奔足•个倌用卡卡号 


validates 一 format 一 of : fieldl , 
: with->/regular expression/ 


检夼 群发邮件不会给 M —个人发两遍 


validates_uniqueness_of : fieldl 


他 ff 1迪否 杷肌群 （muscle group ) 给拼写 
.付 


validates 一 inclusion 一 of : fieldl, 
: in=>[vail, val2 / ..., vain] 


赖絲甚 A " I •以钕 人数搪 痄屮的! 

•m ' 







对十这个系统我们 H 需要检 査儿件事情， m 是 ih 我们石 # 还柯哪些吋 ft. 
的验 hE 器。试试？ 5 你足否能够把下列验沾器和们的用途连接 起来： 


validates 一 length—of : fieldl) 
: maximum=>32 


检迕群发邮件不幺绐卩 ii 一个人发两 ia 


validates 一 format 一 of : fieldl , 
: with=>/regular expression/ 


他把肌群 (muscle group ) 纷拼 
对了？ 


validates uniqueness of : fieldl 


validates_inclusion^of : fIeldl,- 
: in=> [vail, val 2 , … ， vain]. 


检舰户哗啊以放 人数麟:中 _ 个二 


验 ii 器很 简簞而贝工作 得很好 


: 现在数据质录 ¥ 常高，健身课程昀数据也不再神味 . 地消失 r 
^\\mm 为她能够 & 伽 " 郎 A 紱的久 • 户， • 坤 I 

者 q tin 、 川冉枳心他们的付燊 • 消失 . 
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回到 MeBay 







验证你的数据 


M 沾 ay 崖生5很奇忮的事惰 

MeBay 的工作人员听说你最近使用了骀 i £ 器，所以他们试荇把验证器加 
人到你曾经为他 ff 1编写的代码屮。 


m m •他们的$试汴戍功 



MeBay 所做的只是添加 f 验证器来检査新增 r 告的所有域都被填 
鸯了，以及确保数字和电子邮件域的格式正确。 

让我们来看一下具体发生了些什么…… 
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MeBay 鉴证实录 


验证器布效，侄是它们沒冇显示 
错误 

有人输人了一个价格空白的广告来看看验证器有什么问题。 

*'-* .*•• . * .. .• . :. • ■- - • • •. 、 

.• ... . - •» ■ . ■■ ■ 

条不带 j 介格的广告被创建了。_ 




m ^ i 


class Ad < ActiveRecord^iBase 

validates _ presence_of : price 

validates presence of rname 


valIdatcs_presencc_of : price 

验证器阻止这个户告 7 皮创建 O - 


_@由于存在错误，记录并没有 
劈存。 L \ 

• • •.*■■■ .... .二 - : 二•- . 

•.. —:二 * : 、- •一.— — 

❹ 

" ~ .. _ _ * ■ • - - - - - - . 

, . I 1 - - ■•- i . *•- 

系统没有让用户返回到表单，而長 
直接把他们送到了位予 /ads/ 的索 
引 S 面 • -• 

/ 


V：- 

什么地方出错 "F? - 

_• a. - . • 
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meba y 应用有什么问题？ 



噠势4承郃在用逆用支穿茱 
备]達.鉤以1成？銥外的代 
( 砝来检 f 抟# 


0 




f G 二料加过它 



0 


- (^笔上阵 - 

、、解答 

1. 如果某个广告被正确输入，人们看到的下 一页会 是该新广告的页面。为什么 
Rail ; ■显示 了位于 / ads / 的索引页面而不是带有错误信总的 New ad 页面呢？ 

.龟*矜! P 龟译序 &笋年 奉:?.筘 f .. 译*李埤 M 寒今和 . 

/«ds/<ld> 

:二::二:二二二二::::二::二二::二 

/«rfs/<bta^> 


2.验证器在健身俱乐部应用中工作正常。你认为是什么导致了 McBay 的错误？ 

谜夯嚷乐部在用 f 逢用友穿乘釗違. 3 Bmc 松^应用14工釗違的。1穿代砝含检资缚沒# $矛它 
in :::::. f £ ! ant 宣义“(戈场: .泛 有 Ci 么焱， 


.4 


IjEEI 

ctfustalMltlt*<.( 0 l*lu£ 

EJIIm 


si 

t 〆 

rill 

m~- .WI*.- / .「 

I h" n I u .u 
| 一： :::: 二二 

Lls . 


圍 

I 


Jii 




J 
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验证你的数据 


如果你褡建 f 8的页 i ， 你 f 要編写 
f 8的错误洎息代码 _ 

当你使用支架生成应用的代码时， Rails 会生成你 需要用 来处理错误的 
代码。但是如果你手 工创达 代码，你就得完全犇你自己了。所以我们 



0表单页面需要显示所有由验证器生成的错误。 


• ■ 

Mcwad 

1 arror proNMt«d thto *d from twinfl Mv*d [D ___ ' 

如料. «*** 


那么应用要做的第一件事是什么？ 
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控制页面流 


控制器 f 要知道 I 否#在错误 

如果用户把错误数据输人到表单中， Rails 需要把用户返冋到带有 
错误的表单 J ：。 像这样的页面流由控制器来处理。 记住， 控制器 
掌管读写哪些数据以及显示哪些页面。 

控制器代码所要做的就&处现 MeBay 应用甩的错误。 K 面是新广 
告被提交时应用所做的 亨情： 


def create 

@ad = Ad.new(params[ : ad]) 
@ad.save 

redirect—to "/ ads /#{@ ad . id }• 

end 


如羃搀这.我们 

不 4 璧 f 宠命 


代码反复做相同的韦情一一尝试将广告保存到数据库中，然后到 
页而来显示该数据。它并不关心保存是否失败……而这是一个大 
问题。 


但是我们该如何告知保存方法失败 f ? 好吧，在 Ruby 中每个命令 
都有一个返 回值。 当保存广吿失败时， @ad.save 命令就会返 M 
false 。 我们可以使用 dad.save 的返回值来决定我们应泫重新 
显示该页面……还足蛀示保存好的广告。 



f ：. 鱿 S 5 ■•广 

为了做到这 一点， 我们需要学习更多的 Ruby 知识 …… 
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代碚泜键翹 



这儿给出了你用来修正页面流的代码。 

_ 它使用了一个你还没有见过的 Ruby 
S 语言特性—— if 语句。你的任务是 
WL 从代码池中选出代码片段，并把 
^ 3 它们组装起来修正页面流。 


if 





如徉失政 


樣板 。 


注意： 代码池里的每 
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如果这样，那么就那样 


代碚泜键拯斛著 



这儿给出了你用来修正页面流的代码。 

它使用了一个你还没有见过的 Ruby 
® 语言特性 if 语句。你的任务 
■ L 是从代码池中选出代码片段，并 
r \ 把它们组装起来修正页面流。 


辟以％ 


^red^ect_to^ … . 



rt^dtr : action-> 


邦么狄 t 护 s 
供板 。 
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试劈 


史新你的 MeBay 代码来包含一些自定义的页而控制。现在如果我们忘记在价格^ 

域中输人值会发生什么呢？ iLHtad Firths 

有 f 蘄。 


New ad 




TM rvtiiv ^ s •ht m 嫌 N 、 _ 



该波 S 杀。 



当验证器运行时，控制器发现表竽数据有问题，它便重新 M 示该 表单， 这样用 
户就能够纠正错误<»模型并没有保存记录，控制器现在正常运作了。 

除了……是不是漏了#什么？ 


你现在的位置 ► 211 











用户需要明确的提示信 s 


我们还要显示错误消息 f 


应用1新显示表单，这很不错 一" -但足接下来呢？ 



为 r 解决 表牟上 的问题 • 用户需要知道什 么地方出错。 他们孟要错误 
消息来告诉他们： 

0哪个域有问题 
O iv 体达什么问题 

f 在次验证器验证失败时，它会保存一条错误消息到模型 中。 但純们 
%耍把这个错误消息迠示到视阁中。这就息味 泠消息 丄发:从校切屮的 

移到视围里。 

哪个接口与校喂对象的绑定蝻紧密呢？ 

表单！衣电对象 奵一个 特殊的 方法. 它能够牛-成•个错误块。这个" 

法叫 做 err or 一 me s s ages: 
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是时候发布新版代码给公众片听听他们的想法。 
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验证器改进了数据的质虽 


MePay . f , 统现在焉起来铤滋润 

现在系统能够正确地报告错误， MeBay 的工作人员添加了越来越多的验 
证器。控制器检査这些错误并汇报给用户。没过多久，系统中的数据就 
有了很高的质鼠，而错误数歐也急剧减少。 



只剩下一件要做的亨情了。验证器防止 T 主要的数据问题，但这些错误只在新 
广告被张贴时产生 I 

广告被编辑时的错误依然没有被报告出来…… 

- 
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重写这段代码使它能正确处理锚误。 


请在下面的虚线处写下需要错误消息显示代码的文件名。 
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还需要处理编辑错误 



def update 

@ad * Ad-, f incMparams idl) 
r @ad.update_attributes(params[ : ad]) 
redirect 二 to ,, /ads/#.{@ad.id}T' 




rtvuJii 


请在下虚线处写下需要错误消息显甲代码的文件名。 


下面是 “ edit ” 页面被提交时运行的伏码。 


CKU)I ■ 




c®fld = Acf.fi^d (pflrflkvts [: td]) 




redirect to V«cfe/#{(®fld.cd)" 


<hl>Editing <%»= @ad. name %></h-r> _ 

<% form 一 for {@ad, :ui:l=> { :~3<: 亡 1011*»> • update ’ }•) do-1 f | %> 

^ <%= f.error_messages %> 

<p><b>Name</bxbr /><%= f.text__field : name %><r/p> 
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验证你的数据 


你的 Rails 工異箱中的工異 

你已经把第 5 章收入 8 中了，现在你已经把使 
用验证器的能力加入了你的工具箱。 


X S . 

疏 w 眞队=> 32 桧杳似 

节赵过个字符长 

参 If 一一作 一 
e ^ prdsslo ^/ 桧查域秸够匹紀正 ®’) 表 e 式 

vaUdaUsj^veiA^ssjjf 和此检杳數招表中 .a •充其 
他记录节耷 fl«Ldi 相同的免 

..., v «ua 桧 t 域有一个给定的值 
f . error _^ esS «g« $ 5 ■•表荦中的糾 
携型的象 i (j#jsflve^^fite_Httiabu.t6s^ \i 存成 ㈣ 迫 ® 
tmc , 夹败的 ii®f aLse 

她 ？ ^=>讀叫值用研 /AwW 
tev^pUJte.WmUrb i _‘:玄染徐出 
一伙 ▲〜=>,_■ ㈣ 板 
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6 建 立连狻 


+ 把它们集合起 



团结就是力置 。 迄今为止，你已经知道了 Rails 的•些关键组成部分。你 
创建 f 整个 Web 应用并 H. 根据你的需要对 Rails 产生的内容进行丫定制。但 
在现实世界里，情况可能更为复杂。请继续读卜去……是时候来创建 
些多功能的网页了。而11，也足时候来 处现复 杂的数据关系和通过编写你 
自己自定义的验证器来柁制你的数据了。 


这是新的一韋 




拿重署雏仿 I 讀 * 


椰子带你高飞 


椰孑航空 f 要一个汀累系统 

乘坐水上飞# {是 岛屿间旅行最便捷的方式，而椰子航空 (CoComit 
Airways ) 就有着整 个飞行 编队。他们提供景区游览、短途旅行和 
所存 本地岛 屿间便捷的往返航班服务。他们的服务很受游客和当 
地居民的欢迎。 * 

他们的航班需求非常旺盛，所以他们现在需要一个订票系统来帮 
助他们。这个系统需要管理航班和座位预订。 F 面是他们需要保 
存的 数据： 

Flight \^r- 达袅批抗作 £ 




id 

integer 

depature 

datetime 

arrival 

datetime 

destine 

string 

baggage_allowance 

decimal 

capacity 

integer 




2⑽心 




intege 


intege 


baggage 
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连接你的表单 




笔上阵 
解答 


为以下需求写出具体的 指令： 


创建一个名为 coconut 的应用。 ^ T>tS<ri scaffalrf 

rails cou>vui.t 苳存玄 

. 查朽金 .迷 €•幼殊 如 •• 

. 为航班数据使用 scaffold 命令。 

ruby s&ript/g t^ratt s&flffoLd flLcjht de^aYtn.rt：datctlvw4. arns/aiidatetiuvu 

destiwatLo^ ： striKu^ ba^a0c_alU)waku^ ： deci»^l (^aut^-.l^tt^ejr 、於 (/, ^ 

1 a 

• 为座位预订数据使用 scaffold 命令。 

ruby &cnft/QC\^cratt scaffold scat -fU^\>\t_id/u^JUQtr ba 09 a 0 e ： declkM.aL y 

. 仅仅为航班和座位数据使用支架会有什么问题？ 

衿躭筠和瑄仿盎鉍迻用友穿含分則衿躭班和瑾 亡坌威一系列 否#. fSi 它们4•: 泛夯 破铂含起柒。 


raiZe 

逑表掌， 


我扪霈要在—个页 w 上芒 J 5 著到坑 
班和座位预打倌真 

如果我们简牮地创违 i 架而没有定制应用，它将难以使用。为了 
HU 了某个航班的座位，用户不得不从 URL 里寻找航班 id: 

个其个托班的座仿.用户不 



我们 累要二 g 显示航班及其座位预订信息, 
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让我们 i 煮座伎支架代码给 J 
我们什么 




它们中有谁能帮助我们产生航班页面吗？ 
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合并页面 

我们需要让预汀表单和座伎列表出 
现在航班页 i 上 

座位列表和预汀表黾这两个生成的页面#起来与我们想要 ft : 航班 
页面上出现的内容非常相似。航班页面的中间部分看起来像座位 
列表，而末尾则像预1了 表单： 



所以我们需要肮班 “Show” 页面包含像座 
位 “ index ” 列表和新唣位预 V ]•表单一样的视 m 代码。 


拷贝那么我们是不是只要将每个表单里相应的代码 
拷贝到航班页面中呢？ 






建立连接 




Laura : 我个知道。 m 迠我们需要 A: 这个 ijf 面甲.出现这咚代码。 
这&设汁要求。 

Mark : 我知道我们志要比座位列衣和 Mir 表箏出现在这个 ！ Wii 
里。 m 这意味荇我们必项 it 这呰代码出观在那儿吗？ 

Laura : 怎么 f 这些代码有什么 H 题 W ? 

Mark : 崦位列衣 和颅 if 表中.实现的 H 全小 M 的功能。我们不 
能够把它们分解开吗? 

Bob : 把它们分解； H 1 ? 你是说分成不问的文忭？ 

Mark : 造的,这徉我们就町以让一个文件匾示座位列表,另一 
个显承颅汀忐箏，然后从主豇面位禽或者调用这两个页 (6 L 
Laura : 哦 就像关注点分离 （ separation of concerns ) 。 

Bob : 啥？ 

Laura : 关注点分离。 就是说你让一段代码只做一伴事汴。这将 
«£容易庚错。 



U»c<ra 


Mflrte 


Bob : 嗯，听起来梃 W •的……但你怎么实现它呢？ 


我们怎样才能拕一个负靣的沟 
容分藓到 A 个文件里哝？ 


局部模板就是一些嵌人式 Ruby 文件，就像模板 
(template) 。 唯一的不同是局部模板使用以下划线 （_) 
开始的名字。 


如*我们能够把一个豇面分解到多个文怍中，这样就吋 
以 u : 代码变得吏容易管理 r 。 m 是我们怎样实现呢？ 


Rails 允许我们把页而的各个片段存 fig 到被 称为局部网页 
模板 (partial page template ) ， 或者电简单一些， 局部 
模板 ( partial ) 的畝独文件中。局部模板就像迠一个输 
出一小部分！ U 面的 Y •例程。在我 ( f ] 的例子中，我们可以 
使用两个局部 揆板： 一个是座位列表，而另外一个用来 
添加新的座位预 if 信息。 
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嵌入式 ruby 



我们现在有 f 三种嵌人式 Ruby ( ERb ) 文件： 模板、布局和局部模板。但 
足它们之间苻哪些不同，以及每一种 ERb 又是如何被使用在其他类型里的 


你可以像使用一堆原料制作一个汉堡那样来使用 ERb 文件组成-个网页 


布局为^系列 N 贝设定 T 统一的外观，大多数会提供 
出埂作&个飪向顶部和底部的 te 准 HTMLti ! 素， 就像 
包典汉堡的小_面乜。默认怙况下，关联 r 指定控制 
器的所有页面 将共# 同:•个布 W 。 


糢板 (template) 

模板适网诉的书要内容，就像汉堡由的填允食物。模板 
与动作相关联。所以我们有一个模板來显承 航班 if : 细偯 
息，而另一个模板用卡 “New flight (新航班>”表单。 


孱部糢板 (partial) 

个模板会 iMmr 多个不|«|的局部模板； fc 违立页 ifti 的主 
要内容> 局部模板就像?义堡的单 f 原料，比如 西红柿 
或者生菜。局部模板允 i 午你把一个复杂模板分解沁多 
个史 小的部分。 它 也允作你把公托的内 容分离出来, 
比如菜单和导航_栏。局部模板可 a 被模板使用，它也 
"作文被布局 A 接使 R !。 




226 第6意 



建立连接 


一大群涂鈸笤全衮戏服的 ERb 文怍疋洋玩“我是谁？ ”的派 
对游戍。.它们将给你-个线索 f ' V 耶 S 试裉据它们所说 
的来猜出它 n 是 i 隹。假定它们所说的部足真的。请在右边 
的空 P 处填写出这些出席者。 

今晚的出 席者： 

迄今为止任何一位你见过的迷人的 ERb 文件类型都可能 
出现！ 


ERb 文件类型 


我包含导航菜单。 


我包含出现在浏览器窗口上的标题。 


如果有人想创建一个新对象，.峩就显示一个表单。 


我显示 联系邮 件地址和版权信息。 


我将标准外.观的导杭栏賦予一组页面。 
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与你的 ERb 们相会 


一大群穿鈸笤令套戊服的 ERb 义件正 ft 玩“我足谁？”的派 
对游戏。它们将给你一个线尜 你将尝试根据它们所说 
的来诘出它们是谁。假定它们所说的部足«的。请在右边 
的空白处玳写出这呰出席者。 

今晚的出 席者： 

迄今为止任何 一位你 见过的迷人的 ERb 文件类型都可能 
出现！ 



这4芍以*:4用在几个池方妗否 面蛘段 

>c 

我包含导航菜单。 

管个 htTML < tltU /> 部分#由 

布晏輋让 Jl 

我包含出现在浏览器 Sr 口上的标题。 

梭板楗用奋； r 阌的劫作 f. 比如••〜 ； w M 。 

、如果有人想创建 一个新 对象，我就显 示一个 表单。 
iii - 个务 《« fe . ©亡它 45® 的一 邾分. 佟也 
也芍以 *44 丟來汊用 

我显示联系邮件地址和版权信息。 


汉 b 文仵类型 




4晏 


找板 




^我将标准外观的导航栏赋予一组页面。 布晏 

车晏控 *•)? 个 s ®#4 c . i 
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£ Rb 将组织我们的贞 i 

我们需要为预 iT 表单和座位列表创建局部揆板，然后嵌人式/_ liii - il * 

Ruby 将会处理航班! ftlfii 并且在好次执行 render 表达式时调用相^"^ ^ ^^^^, 

祝含 起來- 


eRjb 锊棵旖棣板认 
祅 C 表辈爸部榉板扣硅 ( 2 列褎要 
砗俅板來设泛残们栌扠 班否甬 

B 00 


铋 务列荅 

□ 匈速-个轉部梅裕 
O 祖務打 吝荜泣加扣灸面中 
匚1 创速一个在佑 列表扃 铈梅粮 
口祖磅佐列吝泫加軔货凾中 



m 


Q 0 0 

%/)/ s/7 flights /2 flights /3 


当 RaiU 收到一个航班信息请求时，它会通过嵌人式 
Ruby 使用局部模板、模板和布局来生成黾一 HTML 
响应。 

让我们先看一下任务栏里的第一件事件——预订表 
单。 


0 

flights /4 


0 

flights /5 


0 0 

flights /6 flights /7 


3 

flights /8 
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局部模板 3 是 ERb 


我们如何劍建预汀表 f 爲部模板？ 

局部模板只是 另外一 种 ERb 文件而已，所以它们包含着与模 W 相同的标 
签类型。下曲是我们的_ new _ seat . html . erb 局部模板的内容。它包 
含了与新座位页面一模一样的代码，这也就意味善我们需要做的 就是拷 W 

app / views / seats / new . html . erb , 然后把它另存为 app / views / seats / _ new . html . erb 。 



我们可能已经把局部模板留住了 “ seats ” 文件夹中，但 
是我们要把它移到 “ flights ” 文件夹，这样更易干调用。 
另一件重要的事件是局部模板以字符_幵始。 Rails 通过 
字符 .来区 分页面投板和局部模板。 
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建立连接 


规在我们 t 要在模板中包宮爲部模板 i 

创进局部 校板仪 仅完成了一芊的工作。我们现在志要修改航班的 show , 
html.erb 页面模板来把局部校板包含作:它的输 出艰。 局部模板与校板 - 
祥，仅 仅只是 -段伪装成 HTML 样 T * 的 Ruby 代码。就像 Ruby 代码可以调 
用其他代时-样，校板也 》 r 以很方便地调用局部投:板。 

那么你如 M 调用.个局部換板呢？通过添加•个 render 命令到杬班 S 面 
中： 


<p> 

<b>Departure :< /t» 

<%=h light.departure %> 

</p> 

< P > 

<b>Arrival:</b> 

<%=h ^flight.arrival %> 

</p> 

<p> 

<b>Destination:</b r > 

<%^h ©flight. destination %> 

</p> 

< P > 

<b>Baggage allowance:</b> 

< %5S h @flight. baggage_allowance 

</p> 

<p> 

； b>Capacity ： </b> 




scflt htwd . crto <)4? 样 


<%=b @flight.capacity »<> 

</p> 心 

<%= render :partial=>"new_seat" %> 

<% =.linK_to 'Edif, edit_fli g ht_patb(@fl^ht, 

<%= lin)c_to 'Back', fllghts_path %>_ 


你不应该在 render 
调用中使用实际文 
件名。 


即使局部模板以_幵 
始且以 .html.erb 结束，但当你以 
render 来调用局部桟板时，这两 
者都必须省略。 


render 调用II:嵌入式 Ruby 处理相应的局部校板汴把它的输出添加到文件的 
这个位置。 

局部模板现在应该出现在航班页面中了。 
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开始渲染 



让我们看看 show.html . erb 航班页面并检査预 IT 表单是否正确陡示。 
如果我们输入•些航班信息到这个系统电然后通过这个 URL 来© -下第 
一个航班： 

http :// localhost :3000 /f lights /1 
我们 ㈣ 如下的输出： 



一个奇怪的错误发生了。在我们插入局部模板之前.航班页面还一切正 
常，那么问题出在哪儿呢？ 
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建立连接 
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局部模板也黑要玆据 


我们 f 要绐爲部糢板一个 seat 变 f 


问题出在 ERb 代码包含了 •个对 @seat 变蛩的引用。为什么这是一 
个问题呢? 

这个文件过去是与 S e a t s C o n t r o 丨丨 e r 相关联的！ K 断模 
板。 SeatsComroller 这样初始化 @seat 实例变镜： 

@seat = Seat.new 

佾足现在这个文件变成了 -个将被 FlightsContmnez •使用的局 
部校板，而这个控制器汴没釭 @seat 实例变1：。所以我 们怎要 
把@5681:改成一个名为 seat 的 W 部变敏： 


r<hl>New seat</hl> 

' <% form__for (@seat) 


at) do |f| 


« 的? 1 用， 


scat. 系不袅 @scat 

• if . 



seatM 称为局部变量垃因为在这个 W 部模板之外没办谁会读取 
或荇改写它。但垃如采需要在外部访问.我们该 4 uH 为 M 部校板 
的 seat 变说传递值呢？ 


234 第6簞 



建立连接 



«部梭板和模板很大程度1:就像 Ruby 的方法或者函数。当一个模板渲染 
( render ) —个局部模板时，它有一点像•个函数调用另一个函数。 




板的易郝的唸圣。 么 I 




render 方法能够接受一个名为 locals 的哈希。在呤希里，你可以包含一 
组通过变里名来索引的值。如 MRails 的其他地//那样，名字使用与符号 
(symbol) -样的表达方式。 

供足我们应该给 sea t 陚 f •什么值呢？让我们看一〗、原先 SeatsController 
所使用 的值： 

def new 


由干表蟓会被用来初始化 seat ， 所以我们只需要给表单传递一个新创建 
的 seat 对象： 


那么这样是否就解决了航班页面的问题呢？ 










0^创逑一个预打表荜曷部模板 

□ 倒 逑一 个在佐列表扃鉀 根板 
□ :} g;J 位列 杏泫加 釗费®中 




• hup //iocilh^ 


；4<5(-5)6*3 o 


8a9Q»g« •llowanc •: 30.0 


再次渲染 


on 


Flight* si 

http 7 /locJii»»o$t J000/*l'9»«$/l» 


当 seat 对象被 iH 确初姶化之后，上一次的错误应该蛘以避免了 
刷新航 班豇面： 


236 第6章 



建立连接 



fu .足衣中 . 心:该如何使 ni 这个 id 呢？犮中.怎柞 >)• 
能提供为域提供一个默认值，而无需询问用 


彻3— 个预 穿舄卽模杨 
梠预 n * 荜溧加釗芡面中 
仓!/逑一个/$ 位列義舄卽權桮 
梠 沒位列 吝溧加 釗贲面中 
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默认值 


笔上阵 

解答 你可以在创建 Seat 对象时指定航班号。在 flights/show.html. 

erb 文件中添加你所需的代码： 

一个華夯魚 

<%= render : partial=>"new_seat”, ： locals= ： M:seat=>Seat.newL |f^l^J^=?(^fti0# : ^L..H ! «>> 

— ~ y \ 


这 ft 代砝來 tlf 

fl|r^vi«ws/fU0hts/sh()w.Htw,L.CKb ; 


斤个辛”利 广 馎入呤 H 來綱生 ㈣ 


<p> 

<b>Departure : </b> 

<%=h @flight.departure %> 
</p> 

<p> 

<b>Arrival:</b> 

<%=h @flight.arrival %> 
</p> 


Ct.^&V\ow.V\t 


yvd.ert 兩乇 
起象在 *: 袁(象 


达个样孑 



<P> 

<b>Destination:</b> 

<%«h @flight.destination %> 
</p> 


4 / 


<p> 

<b>Baggage allowance:</b> 

<%=h @flight.baggage_allowance %> 
</p> 


<p> 

<b>Capacity:</b> 

<%=h ^flight.capacity %> 
</p> 



<%= render : partial=>"new_seat", : locals=>{ : seat=>Seat.new( : flight_id=>@flight.id)) %> 

<%= link_to 'Edit *, edit_flight_path(8flight) %> | 

<%== link 一 to * Back*, flights_path %> 
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代碚迆键拯 



<hl>New seat</hl> 


用户不应该需要输入航班号，所以我 
们需要通过一个隐藏域来存储航 
班号。你能够组织下面的代码片 
段来实现吗？ 


<% form for (seat) do I £i %> 


<%* f.error_messages %> 


^ . 

铉藏嘁的禽 <P> 

<%= f. label : name %><br /> 


O 刪 ㈣ 心 



<%= f.text_field : name %> 

</p> 

<p> 

<%= f.label : baggage %><br /> 
<%= f.text_field : baggage %> 

</p> 

<p> 

<%= f. submit "Create** %> 
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隐藏 ， —u 
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你璉_个莰打 名耷曷 砂楔桮 
制逑一个廒佑列表舄初堪核 


试劈 


m ： . '*1 我们宋到炕班琢峽)_梦* 巳绰 从灰申-王消秀 … ii : 如 m 
找1〔1沿0的那徉。 


HiQhn 办 

D „. rt u-. ； .'0<»u n» w°° u,c 

丨一 ， : ^HU. U-.S OOUTC 

OM » •挪 S1 Cut?Wm 

■.««•«• •«>«»«« ; 100 . . ^ * 


C*|>acKV- - 2 

New seat 




M 夂 ** 

_ ' 紗导 


系 4(6 ㈣ 外移获奴 

i；i 的 


I - MnrarTnWi^kfi 

JJU ______ ■■ ll|lllgrr "^^^^^^^ 


印 Qht 
~»m« : 


••OB*®#; 22 0 

^IQitck 


fM ■ 


这个表单一切正常！ 

杭班号现在 ft 动地从航班对象中获取 。 F •步 
做 什么？ 


下 f •我 f}f f 釔澧一个瑄治列 
彖易 钚核杈 
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从模板开始的局部模板 



- id , ©婷 它妖•奋杖 


<%^ render : partial=>"seat_list", :locals=>{ : seats^>Seat.find( : all)} %> 

ii 含注<1唉 •： 让裁们巷 
看…… 


我们还 f 要为座位列表做一个爲部模板 


我们可以使用与我们转化预订表单差不多的方式来转化座位 “ index ” 列 
表 —— 拷贝原来的座位模板文件到一个局部模 板文汴 中。我们把这个新局 
部校板命名为_ seat list . html . erb ： ^ 


但是座位列表爲部模板 f 要一个座 
位数组 

座位的 “ index ” 页而 lii 示 了 -个名 Xj @ seats 的 SeatsController 实例 
变竜:的内容。 ScatsController 是在 index . html . erb 被显示之前创逮 
了这个实例变董。但现在会怎样呢？我们把 index . htrnl . eirb 模板 
拷贝到 f 一个在运行 HightsCommller 之后才显示的局部模板……所 
以我们 就没柯了这样-个包含座位数组的@36扣5实例变 S 。 

这就怠味葑我们儒要为新的_ seat __ list . ht : ml . ei : b 局部模板提 
供一个痄位数组。那么我们应该为这个座位数纟 il 提供什么柞的数据 
呢？下面娃 SeatsController 初始化 @ seats 的过程： 

def index 

@seats = Seat .find (: all "} 

现在，让我 fN 按照类似的力法来调用座位列表，然后 TiTi 它是如何运 作的: 


: _ 的总部-—^ r . c 


views 


'L> ^ _seat_ list.html.erb 


漏 seats 
U_ ind 


index.html.erb 


#( P . 

衿 /f seat Ust.VitkuL.crb 
來舍) •連«拆«板 


我们犹 ( i 个決用添 扣约 

ws / fU ^ ts / sWw’U 夂 


残们把 ii 个作妁砝6啟汨的 
化瘃 狼. 
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建立 



试劈 


党成所心修改，添加新的局部模板，然后检验一下这个应 叫。 


^ 个部分由 scatJ 
ftmUrfc 忐郝痒叛 


UiiCtaifa'tti 
Edit O»»tro« 
lOil PK^LfO. 
Edit Pei ： 'o» 
Uil &»«'»» 


任务列表 

倒逑一个预芽曷部棋栝 
0" 梠预 n * 穿泫加釗负®中 
0^釗3一个攻位 列表晷 部模板 
r ^ 犯沒位列表泫加釗 负面中 


这个丧中.右 起來 ' Wf .常，让我们看看用户是怎么想的。 


你现在的位置 




麻风病人，谁 : 


人们最终登上 了锗误 的航班 




鉍个人都觉得这个系统 看起来 + 错，所以这个系统就正式运行了。不 
幸的是，没过多久就有人报告错误了 …… 


究竞发生了什么？ 

航班 M 面迠示 r 所有 胱班的所有预 iJ ■ 的座位 ! 


c»p«« 


PhQht Name 8««0*0« 

I Sturyeu'i 27 0 Stiow UL! 

Aw ， i，o SlVtW 

»Oouriin* U.O S'W»< g»t b»M* 0 V 

r* Bi4k« 19.0 Shaw iski P«ttfDy 

>nt> ttrdmoi 29 .6 Sucm Etft Pg«»o» 

a^t«9il 10.Q 尹 Hqy Eat DBrtrov 

b'Cjim IS.<3 SNa ， EiM Panttpy 


Cflil 0 !>»Vh* 
CflJ D«H^o»- 
ESblDtfirjU 
Uil DtMfOv 
Ea-i DnLfQ« 
SSii PMtfoir 


^is6>n 
m Carr 


&tl： 

LU ： 


这是怎么回事呢？问题是由 render 命令引起的,它调用了 座位列 
表局部模板。记得我们是这样调用局部校 板的 : 



<%* render : paftialesKseat—iist"*. 

: locals=>{ : seats=>Seat.£ind( : all)} %> 

这会显示数据库里的所有座位列表。当座位列表处在为座位数据设计的 
index 页面时没有 问题 … • 但是视在我们是根据航班来显示数据，我们需要 
限制 显示 的座位使得只 有属于 当前航班的座位才 呌以被 显示。 

我们可以只修改查询器 ‘••… 但是吏好的方法是创逮关系 (relationship )。 
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建立连接 


兵系拕不同模型连捿起来 


你经常会发现特定模型对象常常被一起使用，就像航班和座位预 VT 
那样。你■要使用-种类型的数据，比如航班号，来在另 -种类 
型中*找相关对象，就像肽班中预汀的座位。 


你可以只使用査洵器来读取相关对象。比如，如果你有一个名为 
light 的肮班对象，你可以这样找到相关的座位 对象： 



话的 象啟碓 * 


Seat. find_all__by_flight_id (@ flight. id) 


m 实际上通过关系来连接两个模型会更简单。 


/id 

L 

一次航班有很多座位。 

<- 

舦班 





^— 它们 奋携 f 售鱿邊趄起柒 y . 

关系使得某种类型的对象成为另一种类型的«性。比如，如果我 
们在航班模型上创建一个关系，让它连接到座位模型，我们就可 
以如下引用一次航班所关联的 座位： 

©flight.seats 

它可以返回与前面查询器一样的结果，但是定义两个模型之间的 
关系能够简化你的代码并减少你为每个模型定义査询器时可能出 
现的差错。它也让你的代码更容易阅读。 

听起来不错，那我们如何让矣系运作呢？ 
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关系看起来 就像® 性 





关系将通过匹配 seat,flight _ id 和 flights 
字段中的数据来把座位和航班表中的数据连接起来 


2008 - 11-11 12 : 


为了让关系能够运作，座位表中的域必须被命名为 flight 一 id ， 

而氐 这个域必须为整型。 kr 一 ' 达基®巧杈筠表中礎的航 

班号缒籴 f 耷 ii 个域柁邊趄- 

有效的关系意味着当 Rails # 到如下代 码时： 

<- 一— ci 卷起皋(象基个备 

@f lights, seats 它衾个 

它 会将賊贼 T 麵賊： 

Seat.find all by flight id(@flight.id) 
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建立连接 


但我们如何定义兵系哝？ 

我们将给予 Flight 模型-•个名为 seats 的额外属性，所以我们需要 
在 Flight 模型代码里定义这个 关系： 



class Flight < ActiveRecord :: Base 

has_many : seats 这鱿 4 英联:欠权班 

end 奄多个 4^ 

has _ many 命令接受关系模型的名称，另外，由于它将被用来 
査找相关的座位数组，所以模型名称是£数。这样 ， has _ 
many 的参数是： seats 而不是 ： seat (后者没有最后的 “ s ” >。 
一旦关系生效，你就可以如下使用你的新 W 性： 



@£light.seats <— 

seats 属性返问一个与航班相关联的 seat 对象数组。 




giaiiiaaiiai 



磨笔上陴 


重写 app/views/f light s/show, html.erb 樸板中的这一行来使 
用你的新 关系： 


<%= render : partial=>"seat_list% : locals=>{ : seats=> 


) %> 
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完成所有的修改，然后重新栽入这个页面！航班! ff 面现 在仅仅显示分 
配给指定航班的座位。所以，当我们査看航班号为 I 和3的贞尚时，它 
frM 有不同的座位列表： 
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现在肮班上的行李出现 rw 题。到达机场的有些人带了太多的 o 
李 超出了他们的肮班允许的®鼠。肮班数据 i 己录丫最大行李艰 
發限额， w 很多朵客都不 a •兴， w 为他们作•预 ir 唣位时已经 佐诉航 
空公司他们会携带多少行辛， m 系统没苻给出任何提示。这个系统 

需要适当地修改来防止人们 ft 预 vr 座位时输人过 进的 行卞陵 在 

他 ff 1还没冇获得 fiSU 好的埯位之前。 


- -*• 


. _^ K ' ■. 

W i 



但布些人笮太多的行李 



1 = 1 

f£ll 

mc-o-oo.0.0 

: 三！ 

I! T °"lr 
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编写你自己的验证器 




在 Rails 中，我们通过验证器 (validator) 来检査数据。你认为一 
下面哪个验证器应该被用来阻止乘客携带过多行李？ 

K 没有。我们需要自行编写。 

I | ~validates_lengthcof 
I I validates_format_of 
I I validates_uniqueness_of 
□ validates_incluslon_of 


我们熏要编写自 8 的验证器 

Rails 提供了 -组内置的验证器，它们能够执行很多简单的测试，比如数据 
是否输入或者数据是否使用 T _ 正确的格式。但是有时你斋要使用一些基本 
验汕器无法实现的检査功能。 

在行李 的初子 M ， Rails 并没有提供 v 々 lidate&_ too— much _ baggage ， 

这样的验证器，也不存在最大值验证器。所以我们需要编写出我们自己的 

验证器 。： 个表 ㈣ USE 的余 

如果你在 Seat 代码中 创建-个名为 validate 的方法,达个方法总是在数 ； - 沒 (2 使用鞾宏 
据被保存或者更新到■数据_‘之 前被楱 型对象调用 : a 



errors.acid _ to _ base <...> 命令把-条消息插人到错误列表中。如 
果有一条错误消息被创建，保存或者更新操作就会被终止，而两户就会回 
到表单来修正错误。 
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建立连接 



编写一个自定义验证器来核实航班预订的行李重量小于航班的限定重量。 
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查询器？还是关系> 





编写一个自定义验证器来核实航班预订的行李重置小于航班的限定重量。 


class stat < ActivcR«cord：：'E 
dcf validate 


以•軚 玲的象 中衿出 ^ 爹啄重你芍以洼 
•过 f •尤器扣杈 H 咢 3 鉍鮏 托的象 


. , . : …": .•… . 

If ba00«0« > (fU0kt_id) .bflggflgc.flUowfli^ 


errors .« dd _ to _ bas « ( " you . Wave too > vtw.cki baQ^a^r) 
tv^d 







推荐使用关系而不是手工的查询器。 

无需使用丧询器来査找相关的肮班对象，你可以定义一个座位 
和肮班之间的关系。但问题在干，我们需要怎样的关系呢？ 

之前创逮关系的时候，我们为 Flight 模型賦『• f 一个新的名为 
seats 的 溪性： 



@flight.seats 

但 我们这次应该怎么做呢？之前，我们冇一个 Bight 对象，我们 
希®知道相关的座位是什么。不同的是我们现在要检査一个座 
位对象，而为了实现这点我们需要知道相关的航班。所以这次 
我们需要怎样的关系呢？ 
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建立连接 


我们 f 要反转兵系 

这次我们需要一个是上次的 相反方向 的关系。给定一个特定的座 
位对象，我们需要获得相关的 航班： 



K 

ff*1 将 Flight .. r 

蘇班 。 

我们需要知道某个座位厲 T •哪个肮班。每个座位应该仅有一个航 
班。你认为应该怎样编写代码呢？ 


我们需要在座位上有如下的 域性: 
@seat.flight 


B 


Rak 代碚冰箱祓铁- 

为了从一个座位对象中获得相关的航班，你需要为模型添加代 
码。但是应该给哪个模型添加哪些代码呢？请使用下面的代码 
冰箱磁铁来填空。 


关系将被定义在 


_模型上，它的命令 如下: 


上面使用了关系的模型中的 if 条件 如下: 


if 


r~F^g ht j I - l • 

EDrnu '^ r ^ 

: flight § 


^^ggage__allowance | 


connects with 


sea^J 


|^ oo ^_ forJ | 


: flights 


baggage 


1 □ 
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反转关系 


□ 


Rally 代碚冰箱糍铁辧考 

为了从一个座位对象中获得相关的航班，你需要为模型添加代 
码。但是应该给哪个檳型添加哪些代码呢？请使用下面的代码 
冰箱磁铁来填空。 


关系将被定义在 



模型上，它的命令 如下: 




上面使用了关系的模型中的 if 条件 如下: 


if 


baggage 科 >^| flight |^| baggage 一 allowance 飞 




^ seat^J 


connects with 


「 loo)C-f or I I : flights 1 _ 

' \r\ 1 : seat : J 



flight 


埒 Cf •的樣 f f 扣 
«(4 个鈑本。 


观在 Seat 糢型看起来是什么样孑？ 


让我们完成针对 Seat 模型的 修改: 


app 


晒 models 
1 ~>圖 seat.rb 
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现在当你试图预 n •—个携带行李超过航班限 t 的座位时会发生什么？ 
试试看…… 


殆讼器扣矣彡合沒一 
公遝了 ii 个行 t 问坻 











射向蓝天 (bullet the blue sky ) 


复习要点 

■ 把你的页面分解成多个局部模板 
会让页面更容易维护。 

■ 局部模板、模板和布局是三种嵌 
入式 Ruby 的文件类型。 

■ 局部模板被用来生成页面的片段。 

■ 模板创建页面的主要内容。 

■ 布局被用来创建页面中标准的 
HTML 包装 （ wrapper ) 。 

■ 局部槙板可以被槙板和布局调 

用。 

■ 局部模板可以被赋予局部变 M 。 

■ 局部模板必须以_幵始并以 
• html.erb 结束 0 


你可以使用 render 函数调用一个 
局部模板。 

关系使我们更容易在其他模型中 
查找相关数据。 

关系像查询器那样工作。 

has _ many 属性返回数组。 

belongs _ to 属性返回单一对 

象。 

你可以通过为你的模型添加一个名 
为 validate 的方法来创建自定义 
验证器。 


我必须把页面分解为多个局 
部模板吗？ 

不是必須，但是多个小文件 
通常史容务维护。 

问： 为什么？ 

如果有错误，在很多小文怦 
中定位出错的代码会比较容务二 

A >还有其他原因使得我需要局 
部模板吗？ 

重用，.如果你有一个‘准的 
菜单或者联系方 A 部分，你可以在 
不同的檨 k 和布局中重用它:， 


问： 



没存蠢河题- 

有蠢河题 


我应该如何从布局中调用局 
部镆板呢？ 

• 使用 r e n d e r 方法，就像你 
在橫板中做的那样。 

1^1 • 那么关系会让表通过关键域 
进行连接？ 

是的。默认惰况下，关系通 
.过连接一个表的 'id 域和另一个表中 
以_ Xd 结足的域表 .实 现连接；峰就. 
是4什么航班表中的 id 能够连接座 
么 表中的 f lighti . d ,. 

那么座位表中的字段名必须 
被叫做 flightjd” 


是的广如果你不这么 
做， Rails 就不知道如何建立 关系。 

flightjd 是何种数据类型有 
关系吗？ 

好问題.它需要是整型，因 
为 Rails 的 id 域使用 整型。 
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翁 


请扩展自定义验证器，使得它也能核实预订的座位没有超过航班的载客蚩。 

【提 示： 所有的数组都有一个叫做 size 的方法，它返回数组中元素的个数。】 


class Seat < ActiveRecord: : Base 
belongs_to : flight 
def validate 

if baggage > flight.baggage_allowance 

errors.add_to_base( M You have too much baggage"} 

end 


end 

end 
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size 方法返回数组长度 



请扩展自定义验证器，使得它也能核实预订的座位没有超过航班的载客量。 

[提 示： 所有的数组都有一个叫做 size 的方法，它返回数组中元素的个数。] 


class Seat < ActiveRecord :: Base 
belongs_to : flight 
def validate 

if baggage > flight.baggage 一 allowance 

errors.add_to_base("You have too much baggage") 

end 

If flt0kit.sc«ts.siz« >= flight.cApa&itij 


crrors.add_t4)_bflsc ( ~TVic fluglit is -fully 


tvui 

end 

end 



没存 g 网题- 

古蠡舍题 L 


问 


嘿 一一 等一下……为什么用 **>=" 这个条件？我们不是在检査是否 
有超过允许座位数的情 况吗？ 

是的，俚是记住，关系像查询器那样工作.当我们评信 flight, 
seats 时，我们是从数据库中读取座位信息。这个验证器检查是发生在新的 
座位預订被保存到敫据库之前.所以当前即将被添加的座位并没有被计算 
在内。这就是为什么你需要 >». 


258 第6章 







建立连接 



完成所打前面各页所说的修改，然 / T ； •冉次检验这个应用。 


现在页面能够正确地列出航班座位信息。但是，如果有人打兗在一 
个满 61 肮班上预 1 了座位会发生什么呢？ 
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椰子航空动起来 


迖个系统在椰孑航空投入运行 

肮线上的生活真美好。游客和本地居民发现他 ffj 可以轻而易 
举地使用这个系统。航班不再出现行李超载和超员预 ir 情况。 
事实上,员工可以更高效地利用时间…… 








建立连接 



你的 Rails 工異箱中的工異 

你己经把第6章收入囊中了，现在你已经把建 
立你的大多数连接的能力加入了工具箱。 



X S 

rt ^ dtr ^ artlal =>" 签系- 

ii^rt^tr^artiaL=>" , 

： 1 0& «1 & =>{：^^=> V«Li ) 来把 
変逢给—个易郝權板 

t ) 宠义的鲶讼器代砝係存 4 名 
的核变方••在董 

crrors .« dc (_ to _ b « sc (...) 釗逢一个抟 

m 

beUm^sJo 定义？认的象约萁父的 
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7 ajax 






命 


减少流量 







人们希望获得最佳的生活体验……以及最佳的应用体验。无论你多么精 

通 Rails ， 对于一些传统 Web 应用你可能也会做得不那么成功。有时候用户想要更 
动态的东西或者能够回应他们的每一次突发奇想的东西。 Ajax 使你能够搭建快速 
的，反应灵敏的 Web 应用，它是专门设计用来给予用户 Web 应用必须提供的最佳 
体验，而 Rails 已经提供 f 一组它自己的 Ajax 库，你随时可以使用。是时候方便快 
捷地把 Ajax 精华添加到你的 Web 应用了，让用户享有比以前更佳的体验。 
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服务器神塞 

椰孑航空布个新的促 销计划 

椰户肮空展开 f 一个新的促销计划：鉍次杭班的蝻后三个座位仅 
售一半价钱! 

但有个问题。很显然，毎个人都想抢到这最后三个座位，所以在 
办理登机手续临近的那最后一两个小时里，朵客们+断地 点占浏 
览器卜.的歌我按钮，希望能够获得-张便宜的机溟。不幸的足，不 
断增 K ： 的流铱给椰子肮空的服务器造成 f 巨大的压力。 



额外的沾求汗致邮 f •航空 n 站变惮 r 。 有太多人迕询即将离港的 
航班焓息，这导致其他用户无法通过这个 N 站来颅汀他们所需吭 
班的座位。椰子肮空需要你来检奔一下这个应用，看看有没有什 
么方法能够减少+断冲击服务器的流摄。 
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页面的哪一鄯 分交化 最饫？ 

N 络流釐的大部分内容来自于 航班详 细信息 ® 面一也就是会列出航班座位 
预 i 了情况的页面。这个页面由 app/views/flight s/show .ht ml. erb 模 
板 、 _ seat _ list.html.ert) 以及 _ new _ seat.html.erb 局部模板生 
成。 + 曲是这个页面的三个主要 部分： 


航班信息 

这包含了航班的行李限重以及最大座 t 
数信息。 


座位预订列表 

航班目前预 i r 的座位都显示在这儿。 


座位预订表单 

用户使用这个表单来实际预 iT 一个座 
位。 

当用户按下浏览器上的重栽按钮时，整个！»面都需要从服务器被请求。 

这意味着服务器需要再次从模板和局部模板生成页面，而这全部的内 
游必须被包装到航班布局中。现在，如*:毎次只有一两个 W •求，这不 
会导致任何 M 题，但是服务器 正被大 盪的处理占领。 

我们有什么方法岈以减妗服泠器的负载吗？ 




你现在的位置 ► 


265 






更新修改过的内容 



用户点击浏览器上的 t 我按钮时实阮发生 f 什么呢？好吧，“®我”告诉 
浏览器再次请求整个页面，这足因为整个页而足唯•可以请求的内转。这 
个应用 n 前并不允许浏览器请求电小的部分。现在的情况是，这个页面唯 
一被关心的内容是 《 ur 座位的列表，但是浏览器只能够通过沾求幣个豇面 
来获得座位列表。 



接下来我们需要做的第一件事情，就是修改应用使得网页被关注的部 
分——座位列表——可以通过单独的请求获得。我们需要允许浏览器请 
求一个能够产生座位列表的特定 URL 。 • 
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1 f lig ^D i :— 1 i fiir gh 一 卜 

TTI-^seat.listl [flight-id I |Tm ght .， eats ^ 
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重新路由请求 



把上面的两段代码添加进 routes.rb 和 
seats 一 controller . rb 0 






你应该如何在 routes . rb 中 定义一 个路由 （ route ) ，使它匹配 
针对 /flights/: flight _ id/seats 的请求，并将其映射到 
seats 控制器中名为 flight _ seats 的动作 （ action ) ? 

*/fLL0hts/ ： fU0ht_Lc*/s«ats*, : actu>^= > 一 swts ，， 


: cov^Crolitr= > 'scots' 


秒 / 


• r _.rb 中现料砂公方。 


def flight 一 seats 


代碚冰筘糍铁斛著 

完成 seats 控制器中的 flight _ seats 方法： 


=^ (params [ id"*^ ]) 

[^renderj ^ partial ^> : locals =>{ : seat 8=> 


end 


coconuts 

^ IflM config 
L-> (g| rout 

— fB 卿 


r^— 2 

I app 

pB c ° 


濟个 i 件。 


ontrollers 


seats_con trol ler. rb 
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讀 ^式劈 - 

设想一下，在 id = 2的航班上 ti 经有些被预订的座位。如果我们输人 URL: 
http://localhost :3000/f light s/2/seats 

我 f 『] 会看到什么呢？ 


oo 


既 p 


http: / / localhost ： 3000/flt9hts/2 / scats 




Listing seats 


Fli^hi 

Name Baggage 

Brad Sturgeon 

22.0 

Show 

HdH Dc&trov 

Kirii Avery 

15.0 

Show 

Edit Destroy 

Drew Bcurlinc 

18.0 

Show 

EliU Destroy 

James Blake 

(JafTiocr Brillan 

19.0 

Show 

E$ill Destroy 

12S.0 

SbQ» 

馳 Dc&tioy 

Ted Bnsbm 

19.0 

Show 

Edit Destroy 

Jesse Carr 

15.0 


Edit Destroy 

Jack Casey 

28 j0 

Show 

Edit Destroy 

Tom Halpin 

15.0 

Show- 

Edu Destroy 

Jack Hampson 

18.0 

Show 

Edit Destroy 

Stew Hams 

17*0 

Show 

Edu Destroy 



我们创达的择由会把/£1：191^：3/2/363£3映射到£14味_^63七5 
动作和 seats 控制器，同时会创逮-个称为 flight . id =2 的请 
求参数。控制 器将査 S 数据库中杬班号为2的座位信息汴通过 
seat _ list 局部疫板生成一些 HTML 并返 H 给浏览器。 

ff - F 右边控 制器生 啤的 HTML 。 你注意到 r 什么？ 

得到的 HTML 并不足-个完整的网页，它仅仅是网豇的一个片段 
( fragment ) 。我们能对它做呰什么呢？我 fTM 、 可能 Itffl 户査 f 5 •这 
样的 页面而 不是转到肮班 m 面，因为这肴起朿•点都不 ft 观。再 
说，用户 " r 能 志要预 n ■—个座位，所以我们也沿膂用户能够停留在 
航班页面上 # 

我们忠要 IU 刘览器请求这个页向部分，然后使用它来更新 整个页 
面的座位列表。 

但怎么实现呢？ 
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浏览器更新整个页面 


浏览器不总 I 多 新 M 页 i 吗? 

当用户点击 “ m 载”按钮时，浏览器请求幣个 nm : 



坏消怠足所有的浏览器都坫这么做的。完整 w 求被间化 ft 了浏览器的 
大帖里，“趣:我”按钮怠味肴-敢新载人整个 N 页”，个管怎么说， 
这就是发生的氷情…… 

但为什么会这样呢？ 

过去，浏览器 只能针 对整个页向来工作。 HTML 中没有什么标识吋以 
让浏览器仅仅请求!；(曲的-部分内容……要么仝部，要么就什么都没 
有。 即使现 住我们能够提供0(闹的片段也不管用。浏览器自分没 W 任 
何 / y 法来济求以及使 w 豇面片段。 

那我们怎样才能绕过这个问题呢？ 

笮运的娃，我们可以使出•种技巧来 ih 浏览器仅仅豇新页面的部分内 
容。这种技巧就是: 

弟矣出 t 黃求. 
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H 什么方法玎认岌送清求？ 

每个浏览器的 人帖 中都有一个 JavaScript 引擎。 JavaScript 允 iT •你修 
改浏览器的常规操作。 JavaScript 能够动态 电改 NW 的外观，它能 
够史新止在显4的 HTML 的内容，它还能够在页而 黾响佐 車汁 ，比 
如按钮按下后的情况。史$要的足， JavaScript 也能够发送不依赖 
T •浏览器的请求。 


似这儿的不依赖足指什么呢？ JavaScript 的确能够☆诉浏览器跳转 
到另一个页面，佾它还能够实现 豇多豇 巧妙的功能。 javaSuipt 能 
够 ftG •台5> 悄悄地绐 Web 服务器发送 iff 求汴读取服务器返问的内 
容。 这呜部不需嬰让浏览器跳转到+同的 URL。JavaScript 能够发 
送儿卜条甚至上钎条的 ft 台请求， 而你感 觉不到任何异常。 i 刘览器 
看起来 就像它正在 鼠示一 个页面那怍。 

这•点之所以•如此酸耍就 ft 子 JavaScript 能够发送后台请求 
(background request), 成各说异步请求 (asynchronous request), 
来获得座位列表的最新版本. 3页面片段被返时时， JavaScript^ 
够使用这个片段来更新显示預订痄位列衣的那个页面 K 域。 



驗議敗 



兒器來 m 术 《i 柳巧工： 

f t 藪 t 个费*.-灸 泰的風在罨 € 


来含 S 洱更茭砬 


使 ^JavaScript 来更新 ’*1 前! J(®i 被称为 Ajax, Rails 提供 f 大适的 
Ajax 内置支 t$ 。似我们应该如问使用它呢？ 
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rails 拥有 ajax 库 


f 先我们霜要包宫 Ajax 库 . 

但我 〖f I 如何让浏览器中的 J a v a S c r i p t 发送异步谘求？这种处理似乎 
It 实上，志要在浏览器里运行很多 JavaScript 代码来发 
送 A j a x 沾求。 这呰代码不仅耑 耍 处现请求的细 )V ,而 ft 它还要使 
川.种与大多数 i 刘览器族容的力.式来实现。自行创逑和调试这个力- 
式村千大多数人来说绝对足 个激梦 ，所以很多 A j ax 应川郎使川敁准 
的 JavaScripi 库来简化这一过程。 Rails 提供了这种内霣的库，叫做 
Prototype 。 

Prototype 库位 Tjavascript 文伴火•中名为 prototype • js 的文汁甩。但 
迠即使这个库被乜含在应 用代码 中， 它也小 •会被 a 动包含在应用^生 
成的网豇甩。为 f 确保 Prolotypc « rH 被浏 览器访 H 到，你 g ? 要在你的 
布局中位含一个对它的引用： 



prototype js 2.^ 4 
jflvastKpts j ： 4 矣中它耷 
RflLls— 起 •皮 勿 


<!DOCTVPE htn,l PUBLIC -//W3C//DTD XHTML 1.0 Transitional//EN» 
••http://www.w3.org/TR/xhtmll/DTD/xhtmll-transitional.dtd 
< htm l x m lns--h t tp://wK«.w3.or g /1999/ X html" xml : 1 叫 " ， ’ 卽 " lan ^" en " > 

'Teta http-equiv-content-type" content=-text/html;charse t =UT F -8» /> 

<title>Flights: <%- controller.action_name %></title> 

<%= stylesheetJLink_tag • scaffold* %> Cj 砝侈 3 押饮次 

<%as j avascr ipt_include_tag 'prototype* %> ^ Vj. «4.WcbvH t U ^ 

(ol f *) - 


</head> 

<body> 

<p style*= n color: green M ><%- flash [: notice] 
<%: yield %> 

</body> 

</html> 




javascript 一 in C l u d e _tag 辅助函数可 以确 保浏 览器从正确的 URL 
卜栽 Prototype 库。 

一旦你把 Aja ) (库安 装到了你的网页里，你就可以开始创 建一些 自定义 
的 Ajax 代码了。 


fSaP 

Uc 


views 



its 


flights, html.erb 
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. 狻 T 采我们 f 要渌加一个 Ajax “Refresh ” 键狻 

Ajax 库使得给服务器发送 W* •步请求变得简单了，但是库本身不可能自动 
替你实现 ft 定义的 Ajax 代码。那么我们需要什么样的自定义代码呢？ 

我们的网络 W 题是由千用户点击他们的浏览器上的重栽按钮，这#致了 
为和其他用户提供服务的服务器变慢。我们可以通过在网 M 上添加 
一个标记为 “Refresh (刷新）”的链接来解决这个问题。这个链 接将仅 
仅史新页面上的座位预1了信息，由干它下载较少的 HTML, 所以会比浏 
览器上的 “Reload” 按钮更快。它也会减少服务器的负裁， ih 其他客户 
更容易访问服务器。 

那么 “Refresh” 链接将如何工作呢? Ajax 完全由 JavaScript 来实现，所 
以我们需要让这个链接产生一个 JavaScript 事件。这个链接的事件将调用 
Prototype 库， U: 它发送一个针对该页 seat _ list 区域最新内容的求。 

当 HTML 从浏 览器返 H 时， JavaScript 将动态地将页面上的 seats 种换为新 
的 HTML。 

代码看起来应该足什么样子呢？ 




磨笔上阵 


这段代码添加一个 JavaScript 链接到 Flight 的 
show . html . erb 模板中。请写下每段代码的作用。 


<div id="seats"> .. . . …… . . ... 

<%- render : partial =>" seat 一 list ", : locais »>{: seats =>@ flight . seats } %> 

</div> 


<%=• link to remote ( ST. 

一 - L — 

"Refresh seats ", 、 

: url=>"/flights/#{©flight.id)/seats" ( 
:method= ： >"get" # tz~~ 

L — 

: update=>"seats "y %> .. 


<%= render : partial »> M new _ seat M / : loc £ ils => { : seat => 

Seat . new ( : flight _ id= ： >@f light . id ) } %> 


……飞 

ats } %> 

，二 i 、; : :• ： ^ 

:•乂. ••： W .*•>. 

V ,- ； V 

. 

— 
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你知道的比你以为的要多 




这段代码添加一个 JavaScript 链接到 Flight 的 
show.html.erb 模板中。请写下每段代码的作用。 


id= ,f seats"? 


戧 n 搾宏 t f i 新的否否郝分,, 


<%= render : partial=>"seat_list” , ： locals=>{ : seats=>@flight.seats} %> 


: link 一 to 一 remote ( k~ • 到 • 埤 二七今七 并辞 • 孕•奉苳 . 

"Refresh seats" , ~ i4 • 考 . 辞孕 . 的 X. 

: url=> M /flights/#{@flight. id)/seats” ， fc - 巧 • 參 v" 

: method=>"get" , kr ~ .这.泰钯.我们 s . 

: updated" seats") %>k~ . 


<%= render : partial=> M new_seat ,, / : locals=>{ : seat= 
Seat.new( : flight_id=>@flight.id)) %> 




当嵌人式 Ruby 处理到 show. ht m 1. e rb 模板时， 它将牛 . 成一个 HTML 
链接，这个链接仵被点击时会调用 Ajax 库： 


段 HTML 




<a href="#" onclick="new Ajax.Updater(’seats ’， '/flights/1/ 

seats', {asynchronous : true, evalScripts:true, method:'get', 
parametersauthenticity 一 token=• + encodeURIComponent('7cb578 
0328778ef35ee9d26689784bba0d562170 , )}); return false;">Refresh 
seats</a> 


<hl>New seat</hl> 
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没有蠢网蕺 

产没有齡\ 


I ®)*什么是异步请求？ 

异步 W 求就坫运行在后台的 W 求。 异步沾 
;R 由 JavaScript 屯成。 

i ®)- 它与普通请求有哪些不同？ 

I •• 作通 ift 求作 HI 广点 , i ;. 链接成芹输 AURI . H 1 
产生_异步请求削在 JavaScript 响应某个事件时 


重载页面真的粟要占用那么多带宽吗 V 

4 ll!R Wlfii 的 Jt •他 部分 m «的 HTML 的 
话就会》而且,浏览器会试 ft ) 重 栽財面 上的围 
片,这也会占用较多的带宽。再加上 K 面的其他 
部分也志要尺 Wt 处理 >)' 能够 被创让 出来。 Ajax»f 
以保持豇 面的 it - 他部分不变,减轻 r 服务器的负 
我。 

I ®) • 我必须要愐 JavaScript 才能写 Ajax 代码明 •’ 

Rails 幺 A 你生成 Ajax 代码，所以你汴不 
番要 r 解 JavaScript 。 但是，如果你情 JavaScript ， 
你就能够吏好地控制 Ajax 调用产生的方式，也能 
够更好地理解你的应用程序足如何运作的。 

l ®)» 生成的 JavaScrip 丨创建了一个名 
为 “ authenticity - token ” 的参数。它是用来做什么 
的呢？ 

I Rails 使引真实性令牌 （authenticity token ) 

来保证清求来 S 〒 Rails 生成的沉面。没冇角实 
性令牌， Rails 将拒绝该请求。 


蠢问题 

令牌是如何工作的？ 

它足个山 Rails 生成的值。包含这个攸 
的请求被认为来自 FRails 创途的页面，而不是 
那叫忒阁访 h 你的系统的第三方应用的！ a 面。 

1^* 你刚才说 Ajax 请求由 JavaScript 而不是浏览 
器发出.但 JavaScript 不就是浏览器的一部分吗？ 

是的 • mikJavaScripdl 笮彳以发 送+ 
同丁泞通浏览_序的 ifi 求 这坫关键点所 
在。 Ajax 请求允许你史新页面的•部分而不需要 
发送整个页面的请求,也不会修改浏览器的浏览 
历史 • 

• 为什么我们要使用 javascript _ include _ tag 辅 
助函数而不是直接输入 HTML 来载入 JavaScript 呢？ 

如*:你想 『 IlLSHTML ， &接的 HTML 也 
能工作，仉足 RaiIs 歼发人« ft 釘可能的怙况? C 
都会尽缺使泔辅助闲数。輔助闲数通常比 HTML 
正文要 fe 小， I fti R 它们会为你填写应用相关的 
配置信息。例如， javascrfpt_includejag 辅助换 
数将把路柃设 苕到你准的 javascript H 淆卜、V 
javascripts/...” o 

1^)* 这听起来没什么大不了的。 

辅助确数还在 JavaScript 位1!的*•添加 
了一个长数值。 

1^)- 这有什么用处？ 

这意味苕如果你的 JavaScript 庳发生 H 壬 
何改变，用户将总是能够获得最新的版本。 
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试驾 



现 ^ JavaScript 按钮已经就绪，是时候来看#我们的应用长什么样 
f 。 歌新我 人你的应用;^检验一下。 


0 第一个用户来到航班页面预订 

座位。 

他能够看到肮班的具 怵位息 .d 
经被 预汀的 座位列表以及 mu* 表 
準.。在座位列表和 war 表单之间 
则是新的 Ajax 按钮。 



o 


第二个用户访问了这个页面并预订 
了一个座位。 

当表单被提交时， 她的豇 面进行 r 
相应的刷新，她能够#到她新预汀 
的座位信息。其他后续来到这个页 
向的人也能够#到。 m 足第•个用 
户呢？ 
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Flightt vhow 

« ► C - ♦ ■ , ft 


第一个用户可以通过点击刷新 
按钮来查看最新的座位预订情 
况 c 

这个按机触发一个 JavaScript # 
件，而这个事件又调用了 Ajax 库 
来刷新唣位列表，敁示新的座 
位预 ir 信息 # 



为计幺我一定荽*击別新 
垵钰采金蚤变化嚷？系统不 
能够 I )动更新达令迈香蚂？ 


系统工作得挺好，但是右些用户在想为什么他们一定 
要坐在那儿不停地点击一个按钮来查看新的座位预 
il* 信息。如果豇面能够在有新的 MIT 发生时 fi 动地£ 
新，那该多方便呀。 
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再铪 k 一再给 






浏览器 f 要主劫询问 E 新 



m . 是要实现&动裝新页而 f ! •个.问题，译个 h 题长 q rWebern ： 
A 成。::在二个完美 的世汴 yi • r % m \ ui 汝倍 i 杆变化， wcb ^ 
用就能畀知 ffl 户。不幸的是， Web 服务器泮不是这样工作的。它 



服务器仅 <m w 请求时才发送响应。如！ ft 服务器冇 々它想 ih 浏览器知 
道的新信息，它做不了仟 何參 情，只能等浏览器來询 m 新饴息。 

这意味着如果我们想 it 浏览器在座位列表变化时被自动告知,那我 
们只会失望而归。相反,我们需要让浏览器不断询问,再问，继续 
问 …… 








ajax 


我们一定要让浏览器反复珣问码？ 

回想一下 Ajax “Refresh” 链接的工作方式。釘人点占它时，链抟 
产生一个 JavaScript 啦件，这个啦件相应地⑽用 Prototype 堆，请求 
K 莪座位列表的新版本。 

关键之处在于这一切起源千 一个啪 怍， 一个发 生在 JavaScript 之外 
的事件。 


亊件 JavaScriptBl^ JavaScript 代格 


(( 






一 ^ JavaScript 可以把自 d 注册到某个赛件±,也就&说当事件发生时， 
相 I.:/: 的 •lavaSeript 就能运行。 

在我们的例子电，我 (N 需耍不断地运行钔 M 的 JavaScriptR 码。那么哪仲 
亊件町以做到这•点呢? W 吧，它肯定不是由用户动作产垂的某个亊件。 
相反，我们需要把 JavaScript 注册到 * 个定时器 •! 以1: I .。 

定时器是•个系统事件，它按照固定时 N 间隔发生，通常为每隔几秒。 
我们盂要创迖•个定时器，然 G •把 “3£ 新崦位列丧”的 JavaScript 注册到 
它上面。 


幸运的是. Rails 可以帮助我们实现它。 
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定时器迸发事件 


你玎认像鉴咁按钮事件一样監咁定 
时器事件 

通过点击按钮来运行-段 Ajax 代码~甸隔几秒钟运行•遍 Ajax 代 
码的唯一 差异在 于你正在监听哪种來件。 

山干这个原因，我(门在模板中放 S 的 Ruby 代码事实|.很像我 
们用来创达 JavaScript 按钮的 代码： 


<%= periodically 一 call_remote( 

: url=>" _ 一 获 Si 扣 /| 佬列表的 

:method=>”get' 不 f 扣我鴉 - 

: update=>" - ' ^ 这 € 我 fHil 在 t 祕的 资面部 分的 M , 

: frequency=>" _ ") %> 

苓个宅的器羃碑二间的蛄數 


这段代矾将创逮•段每隔几秒就发送 次 新座 R 列表请求的 
JavaScript。 然后它将根据从服务器端返 M 的 HTML 来 史新豇 面的 
指定部分。这个辅助函数与创违 JavaScript 按钮的那段代码的唯一 
差# 在于： 

O 按钮需要题文字。 

o 定时器耑要被给定•个频 
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真的没有办法让服务器联系到浏览器吗？ 

浏見器可以维护一个与服务器之间的开放式 
连接 (open connection ) , 仅这 意味着即使那些很少 
用到的应用也会需要大量的连接。 轮讷 ( poll ) 服务 
器是史常用的方法。 

定时器的频率总是以秒为单位吗？ 

是的，頻牟总是以秒为单位。这看起来有些 
古恪，它被称为頻率但并没有给出頰率（比如丨分钟 
内触发多少次）。相反，它给出“周期”，也就是两 
次触犮之间的时间间瑀。， 


默认频率是 多少？ 

默认情况下，頻率是10秒。 

t 1 ®)' 局部页面的 id 从哪儿来？ 

HTML 中的每个标签都可以蚨予一个 id 。 它能 
够唯一标记押页的特定部分，通常， Ajax 应用把局部 
页面包敦在一个带有 id 的<<1以>标签内。这徉你就能 
够一次性地为单一标签或者一组 HTML 标签賦予 id 。 
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点击刷新还是周期性事件？ 


笔上阵 - 

解答 航班页面依然需要包含刷新按钮，但是它也需要定时器代码。 

编写定时器代码并把它添加到 app/views/fl ight s/show, 
html.erb 中，使得座位列表能够每分钟自动更新 3 次： 

<^= ftricdlca ILij Ll_ rey^ott ( 

: nrl=> ， 

：\^，cthod = > “gtt", 

: ucpd«te=> "seats", 

: = ， "so" ) J > 
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o 


o 


o 


现在定时器代码 Li 经被添加，系统砹该能够 ft 动 更新嵴 位 surfri 息而无需用 
户干预。 


第一个用户来到航班页面打算 
预订一个座位。 

他能够看到航班的详细信息。 


当他正在预订座位时，第二个用 
户访问了这个页面。 

她很快就预订了这架航班上的一个 
座位并提交了数据。 



— 0,0 , 〜 

蟎 : C [ < , * J * J 


FmM 

r«ntcn*M 


UitenuAx 
Sm EdttaiBfW 


«，an 1 * o sum Ukt 

Con*rd IS 0 Srwm iflltfiUUAX 

Ovirf，* Conixk ； 9.0 S"o» MU Dwtlran 


New seat 


第一个用户自动看到新的预订 
信息。 

即使第一个用户没有动键盘，页 
面也会在20秒内自动更新座位列 
表。 

免砝 S 嫌^ 



New seat 


E(Ht Destroy 
Q^stroy 
Edit Ott^ov 
Etft Destroy 
Edit Peatfov 
Edit Destroy 
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ajax 真情指数 



真惟祎歎 

本周访谈： 

应用 Ajax 来加快脚步。 


Head First: 你好， Ajax , 欢迎欢迎。很布兴 
你…… 

Ajax: 我也很高兴。 

Head First:. 参加今 天的访 谈。 

Ajax: 哦，我刚刚打断了你。 

Head First: 那完全 . 

Ajax: 我经常这么 f 。 +好意思。 我吋 能有呰，你 
知道的，超前。 

Head First: 你是•种忙碌的技术？ 

Ajax: 你#到了？我刚刚电新了数据表！什么 
技术？我不属于技术。我是一种生活方式，伙计！ 
或者说至少我是一种编写 Web 应用的方式。 

Head First: 什么意思？ 

Ajax: 好吧, Rails , JavaScript , Prototype 这些 
家伙是软件。这么说没有任何问题。这很酷，但是 
我在它们之上。 Prototype 只是一个实现了我的支持 
库. 

Head First: 那你是什么呢？ 

Ajax: 我是-种设计技巧。当你发送一个异步 
JavaScript 请求来更新-个网页时，你就会用到 
我。 

Head First: 异步?那就是说你的请求…… 

Ajax: …… 打断 r 通常的浏览器处理，足啊。请求 
在后台产生而用户还停留在3前！ U 。 


Head First: 几乎任何事怍都能触发 Ajax 请求？ 

Ajax: 足啊。 XHR 能够被几乎任何 任何一种 

JavaScript 事件触发。 

Head First: 你能再说遍吗 XH …… ? 

Ajax: XHR 。 不好意思。 XHR 足我的 Ajax 请求的小 
名 。 “XML HTTP 请求 （XML HTTP Request ) ” 
是正式的名字。 

Head First: 你说你不是软件，但人们确实安装了 
Ajaxl $, 不是吗？ 

Ajax: 你可以歃头开始编写你 自己的 代码， m 是 
的确，大多数人会使用 Ajax 库，比如 Protoype 
库。 Ajax 库可以创建请求并处理返冋的内容。 

Head First: Ajax 请求返冋什么样的数据呢？ 

Ajax: 各种各样，五花八门，伙计。 HTML 格式的 
页面片段， XML 格式或者 JavaScript 格式的数据， 
甚至是 JavaScript ft 身。 

Head First: 我明白了。再多说一些吧， 

JavaScript . 

Ajax: 那是啥？哦，不好意思，老兄——要走了。 

Head First: 什么? 

Ajax: 有人 刚刚点 ili 了一个 JavaScript 按钮。这个点 
击事件提到 f 我的名字。冉见…… 

Head First: Ajax ， 谢谢 . 

Ajax: 不客气。 
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布些人的 f 身派对有问题 5 


我襄为我的单身泜对賴订 19 
令佴是我不得不反复点 
击 lack (后邊 )" 键采逄©到 
航班克 Sf 


3你 《 nr —个座位时，浏览器提交表申-给服务器，然后浏览器跳 
转到-个显示了预 i 了的座位的页面 # 但是如果有人需要預 IT 一批 
座位呢？这种情况 r ， 他们不得不点 il ? 浏览器 h 的 “ Back ” 键来 
返问到航班豇面，然后预汀另一个座位……接猗获得另一个确认 
信息 • 再次点击 “ Back ” a 


目前我们已经编写了无需跳转到新豇面便能史新座位列表的代 
码。那么我们能够为座位预 n * 实现戈似的功能叫？ 如果表 单能够 
发送预 in # 求给服务器然后更新座位列丧，那么用 户就可 以保持 
在同一页面。如果他们需要颅汀另一个座位，他们就可以 a 接在 
当前页进行预订。 




/ I 话於象扣择 J 象輩邾 
s 孑在軚筠负面上 
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使用 javascript 发送请求 


表单 f 要生成一个 Ajax 清求 

我们知道，如裝让浏览器提交表单，我们会波送到另一个贞曲' 
去。这就像我们前面遇到的当用户点 * 浏览器的 “ Reload ” 按钮 
时所发生的问题 —— 这是浏览器的内置行为，我们无法改变。 

我们应该怎么办？ 我们盂 要使用一种不 N 的表笮。这种情况下 
我们小能使用标准的 HTTP 表竽，我们志要使川 JavaScript 表中， 
然后 U ： 它来产生•个请求。 




Ajax 请求 


与只是要求浏览器提交表单数据不同的足，我们需要提交按钮 
生成一个 JavaScript 事件，这个事件将使用 Ajax 请求来提交表 
单数据。为什么这一点很秉要呢？因为这样的话，预汀座位的 
动作就不会导致浏览器切换到其他页面。 
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表单 f 要由 JavaScript 來 控制 

所以我们需要将简的的 HTTP 表电 转化成能够中成 JavaScript 餐怍的 
m .， 间时这个表取还志要动态更新3 前豇 而，而不是让浏览器跳 
转到不 MURL 。 Flfii 足 MU * 灰单局部模板的内对： 



但我们怎样才能让表单以这样完全不同的方式运作呢？我们需要修改 
这行代码： 

<% form_for(seat) do If I %> 

成这样： 

<% reinote_form_for (seat, : update=>' seats' ) do I f I %> 


这个改动很小，但是这种改动的背后却是表单将以非常不同的方式运 

作…… 
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还记得控制器吗： 



当用户转到航班页面 (http://localhost:3000/flights/2> 时，預 i 丁表单看 
起來和以前没有 不同： 



当然，搭后的 HTML 有很大的差那么当一个新 
的座位被 fmr 时会发生什么？ 


Rights 讣⑽ 

ioOQ^'9"« 


有些不对。检査-下你的数据库……座位被 
正确颅 irr , 但迫航班页面显示错误，为什么 


我们修改了视图:中的代码，但是控制器代 
码 服务器端的代码 依然在做符与以前 

样的水怡，它返回 r 新 war 座位的洋细信息 
的 HTML 。 而我们所霜要的足座位列表的婊新 
仿息 • 




让我们来修改控制器 代码。 
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我们 f 要管狳 create 方法 

seats 控制器屮现有的 create / j •法如 K 所示： 


- - -- " 

--- ^你不涑楚 d 杳代砝4緻0么的也沒关系, 

def create , m=!l _ seatl> < . i 正肋料軚含穿如⑽。 

@seat = Seat . new ( paramst . seat !) 

respond_to do I format I 

if f @ rnu = 

Eormat.html l redirect_to(@seat) ) : created, 

format.xml f render : »nl - @seat, ^atus 
：location -> @ seat ] 

el fLmat.htmr ( render : action => " neW * } 

1 rend ：： t r ：. 



我们需要把它替换成创建 Seat 对象的代码，保存对象到数 
据库，然后渲染一份座位舛表的新拷贝。 m 足这样的代码 
应该如何实现呢？ 


seats controllerrb 


身笔上阵 


编写一个新的 create 方法，这个方法将总是根据表单数据来创 
建 Seat 对象，接着把新对象保存到数据库中，然后渲染 seat _ 
list 局部模板的内容。 
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我们需要部分响应 




SMt 时象 


编写一个新的 create 方法，这个方法将总是根据表单数据来创 
建 Seal 对象，接着把新对象保存到数据库中，然后渲染 seat _ 
list 局部模板的内容。 

create 

i . i^«W (pfl rfl tots [:SWtl) 

不用筘心•係溽 15 •威劝. 


〒 4 你 ^ ^ ^^^ v.r ) r : partl«L=> fli^his/^at_U^t' , : U>&«U=> {: scats = ht.seats} 

……⑽ . 


迖段代码 会布怎 样的效果哝？ 

新的 create 方法表明，’1 Ajax 表单提交一个新的预 if 时，它将从 
服务器那儿接收一份痄 R 列表的新拷 W : 
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Flight 

Brent Otdee 
Tom Ch^'SUt 
Ryan Cleary 
luhen Col lord 
CKo/ltfe Co!».o* 


Name 

19.0 

ISO 

19.0 

16.0 

19.0 


Jesse Jame» C«rretM2.0 


« no 


.< £► C , < „ ocaihoM 100 ..‘ 





New seat 


Nait»e 




试劈 


假定•个 m 户来到航班豇曲件提交 r 一个新的座位预 ir 请求: 



系统工作了! 财•当 •个预汀发生时，谢览器 
将悴留在当前页，而 r 新的 mma 录也立即 m 
现。 


New seat 

Name 



o 





EE 

丨 EdtlcEdiscdtti 迦 3 
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休息 … j 


复习要点 - 

■ JavaScript 能够给服务器发送后台请求。 

■ JavaScript 能够使用返回的 HTML 来更新部分页 

面。 

■ 使用后台请求来更新页面被称为 Ajax 。 

■ 这种请求被称为 XML HTTP 请求 （ XHR ) 。 

■ JavaScript 可以在事件发生时运行。 

■ 率件可以是用户动作（比如鼠标点击）的结果 
或者系统事件（比如定时器）。 


如果你不希望表单导致浏览器跳到新页面，你 
需要把它转化成 Ajax 表单。 

你需要把 form _ for 改成 remote __ 
form _ for 来使得 一个表 单变成 Ajax 表单。 

处理表单诮求的控制器代码能够返回用来更新 
页面的 HTML 。 

如果你给表单一个： update 参数，它就会知 
道在页面的哪一部分来放置返回的 HTML 。 


问, 


问 


-魅 


有蠢河 骚一 


为什么我们只需要修改表单 
辅助函数而不用涉及表单中的所有 
域？ 

夂单中的域可以保持不变是 
因为它们和以前那榉仅忟包含敫据 
域 . Ajax 表单和 “ 常规 ” HTML 表单 
味一的差别在于 Ajax 表单的 onsubmit 
麥怦调 ffl Prototype 库而不是提交这个 
表单其他地方没有区别。 


问： 


我看到其他地方的 Ajax 表单 
由 •• form _ remote _ for ” 生成。这有 
区别吗？ 

• 没有 form_rem(>tc_for 只 
是 rcmotc_form__for 的别名。它们实 
现的功能是一样的 . 


如果我要转化一个未绑定的 
form_tag 呢？ 

答： 有个 Ajax form_remote_tag 可 
以被用来作为替换 .. 

1^)* 我不明白。“表单“可以替 
换” 页面” 中的 HTML ? ? ? 

不是这徉。表单调用 
JavaScript 函数来产生一个 Ajax 请求 
是 JavaScript 而 跃替换 了页面中的 
HTML 。 


问, 


当服务器收到表单请求时， 
这个谞求还和以前一样吗？ 


答: 


这个请求就与它是从 
个 HTML 表单中犮送出来的 


# c Prototype 会构造这个请求让它 
看起来就是一个完美的常规 HTML 请 

求。 


问： 


Ajax 表单使用什么样的 
HTTP 方法？ 

就像 HTML 表单那徉 . Ajax 
表单默认使用 POST 方法， 


问： 


但我能够修改这个方法，对 


你可以通过在辅助扁数中提 
供一个 : niethod=> 参数来修改 POST 方 
法。 
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我是堆? 

一大群穿戴笤全套戏服的 Ajax 俱乐部成员正在玩“我是 • 

谁？”的派对游戏。它们将给你一个线索，你将尝试根据 
它们所说的来猜出它们是谁。假定它们所说的都是真的。 

请在右边的空白处填写出这些出席者。 

今晚的出 席者： 

迄今为止你见过的任何一位迷人的 Ajax 成员都可能出现！ 





我是 Rails 用来从浏览器中产生 Aja ) (请求的库。 

我是运行在浏览器中的语言。 

我是 Ajax 应用程序中使用的请求，我的朋友叫我 XHR 。 
我是一个亊件，但是我不是用户亊件。 

我被用来根据对象生成 Ajax 表单。 

我能够调用注册在我身上的浏览器代码。 
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复习 ajax 


一大群穿戴着全套戏服的 Ajax 俱乐部成员正在玩“我是 
谁？”派对游戏。它们将给你•个线索，你将尝试裉据它 
们所说的来猜出它们是淮。假定它们所说的都是真的。请 
在右边的空白处填写出这些出席者 P 

今晚的出 席者： 

迄今为止你见过的任何一位迷人的 Ajax 成员都可能出现！ 



我是 Rails 用来从浏览器中产生 Ajax 请求的库。 Prototype 

我是运行在浏览器中的语言。 

我是 Ajax 应用程序使用的请求，我的朋友叫我 XHR。 

我是一个事件，但是我不是用户事件。 4 

我被用来根据对象生成 Ajax 表单。 rtt^ott_forv^_for 


我能够调用注册在我身上的浏览器代码。 


拳4 
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航班预汀布个问翅 

当单旮派对组织者碰到这个问题时，他正在预 n ■他的单身派对之旅。他 
开始 Mir 座位的时候还有好多空位，但是接下来…… 




这听起来很简单，但你能够看出什么问题吗？ 
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同时存在的异步 请求？ 

我们 H 知道如何一次更新页 i 的一部分 
沟容 

到目前为止，当我们发送 Ajax 请求时，我们总是仅仅使用服务器返问的 
HTML 吏新页面的一部分 内容： 


秦求 javascript 



邡么迖次布什么不间哝？ 

这次的差异在于我们需要同时更新座位列表和页面顶部的通知 
区域。它们是两块完全独立的、需要被替换的 HTML 片段。 

那么我们如何使用服务器端的单一响应来实现针对页面的多处 
更新呢？或者发送多段 HTML? 

实际上有更简洁的方法来实现基于单一请求结果的多个操作。 

这儿的技巧便是在响应中返回一些不同于 HTML 的东西。 
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控制番 f 要返设 JavaScript 而 
不是 HTML 

如果控制器返 I "] HTML 数据给浏览器， JavaScript 通常 R 是简 
中.地处理它，比如用它来替换页面的某一部分。但是如果控 
制器返问 JavaScript 给浏览器，那么无论控制器需要让它实 
现多少私情，返回的代码都"了以完成。 





所以，如果拕制器希望更 新豇面 h 的座位列表， ft 接显示 • 

条确认消息，然后执行一些花哨的动画来让整个页面颉倒过 

来，所有这些所需做的只是返回恰当的 JavaScript 代妈访名的不含想！ 
论 JavaScript ^ 含什么，它都会被执行。 {i 
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rails 懂得 javascript 



你可以让 Rails 为你编写 JavaScript。 

如果控制器需要返回 JavaScript 而不是 HTML， 你可能以为你需要了 
解如何编写 JavaScript 代码。但实际上你不需要。 

Rails^ 供了一个叫做 JavaScript 生成器的对象，它能够完成正如它 
的名字所提示的功能 它生成 JavaScript。 




事实上，虽然馑得 JavaScript 大有益处，但是大多数时候，你返回给浏览器的 JavaScript 代码实现 
着很多标准化的事情，比如替换一段 HTML ， 或者隐藏豇面的某一部分，或者调用 - 些 JavaScript 
库函数来实现…个动_。而 JavaScript 生成器能够编写代码来为你实现这儿的每一件事情。 


你所要做的仅仅是正确地调用它。 
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代碚冰箱磁铁 

完成控制器代码以生成 JavaScripU 用来替换 • notice ' <咖/>中 
的 HTML 来表明座位被成功预订。 


def create 

@seat = Seat.new(params 【： seat 】 〉 
render : update do |page| 
if 

page. 

else 


page. 
end 
end 
end 




I - 1 1 ? notices^ 

I replace htng | ‘ 



• Seat was successfully booked' 
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编写代码来生成代码 


代碚冰葙糍铁斛著 

* 完成控制器代码以生成 JavaScript ， 用来替换 ’notice' < 心/>中 

的 HTML 来表明座位被成功预订， 


def create 

@seat = Seat.new(params[ : seat]) 


render : update do |page| 


if 




page. 


replace_html | 


'notice 




’Seat was successfully booked' 



else 

page 

end 


repla ， 
• ^_ 


ce html 


• notice' 


],|_Jj orr y ~ the seat could 


^ot^be^booked • 


end 

end 



控制器代码生成 f JavaScript 来更新网页中带有 id= ‘notice’ 的区域。 
那么网页中的这块阮域是哪儿呢？好吧 肮班贞 面的布局包含厂一 
个位于毎页顶部用干通知的特定输出区域。你需要修改航班布局并为 
< F » 元 素添加-个 id: 



300 第 7 章 








ajax 


那么 Rails 生成 了些 什么? 

JavaScript 生成器创建了如下的 JavaScript : 


叫 Wte 吻 


个. 


try { 

Element.update("notice", "Seat was successfully booked"); 

Element.update("seats", "<hl>Listing seats</hl>\n\n<table>\n 
<tr>\n <th>Flight</th>\n <th>Name</th>\n <th>Baggage</ 
th>\n </tr>\n\n\n <tr>\n <td>Brad Sturgeon</td>\n <td>22.0</ 
td>\n <td><a href^N"/seats/l\">Show</a></td>\n <tdxa href=W 

seats/l/edit\">Edit</a></td>\n <td><a href-\'Vseats/l\ w onclick-\"if 
(confirm('Are you sure?')) { var f = document.createElement('form 1 ); 
f.style.display = 'none *; this.parentNode.appendChild(f); f.method = 

* POST'; f.action = this.href;var m = document.createElement('input'); 
m.setAttribute('type ', 'hidden'); m.setAttribute('name ', 

method'); m.setAttribute('value *, 'delete*); f.appendChild(m);var s 
=document.createElement('input'); s.setAttribute('type *, 'hidden') / 
s.setAttribute('name 1 , 'authenticity 一 token'); s.setAttribute('value *, 

»aec87b235224924109e33b3207d464c64207e733 , )； f.appendChild(s) if. 

submit U ; }; return false; \*'>Destroy</a></td>\n </tr>\n\n <tr>\n 
<td>Kirk Avery</td>\n <td>15.0</td>\n <td><a href*\ M / 
seats/2\">Show</a></td>\n <td><a href=\"/seats/2/edit\ w >Edit</ 
ax/td>\n <tdxa href=\'*/seats/2\ w onclick^X^if (confirm('Are 
you sure?’}) ( var f = document.createElement(* form'); f.style. 
display = 'none'; this.parentNode.appendChild(f); f.method = 1 POST *; 
f.action = this.href;var m = document.createElement('input'); 
m.setAttribute( 1 type', 'hidden*); m.setAttribute('name', 
method'); m.setAttribute('value*, 'delete'); f.appendChild(m);var s 
=document.createElement( 1 input *); s.setAttribute('type', ’hidden*); 
s.setAttribute('name', 'authenticity_token'); s.setAttribute('value', 



这段代码会在 Ajax 预 汀 表单被提交到控制器时被返回给浏览器。以 
前，浏览器将获取控制器返回的内容，并用它来更新奴面的某一区 
域。但是现在，我们让浏览器执行返回结果。我们希望它运行我们 
的生成的 JavaScript 。 


但我们如何告知表单执行返回的 JavaScript 响应? 
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我们完成了吗？ 

如果你沒冇表碘在哪儿故 I 响迨，那 

么它就会狨故行 

让我们看一眼牛-成 Ajax 表单的嵌入式 Ruby 代码： 


<% remote_form_for(seat, : update=>'seats') do |f| %> 
<%* f.error_messages %> 

<%= [hidden 一 field : flight^id %> 

<p> 

<%= f.label : name %><br /> 

<%» f.text_field : name % > 

</p> 

<p> 

<%■ f.label : baggage %><br /> 

<%- f.text 一 field : baggage %> 

</p> 

<p> 

<%- f.submit "Create" %> 

</p> 

<% end %> 


这段代码创违广所有需要在表黾的 “ Create ” 按钮被点击时触发 
Ajax 请求的 JavaScript •接 F 来，这个表单接受服务器端返回的任何 
响应，用它来 替换豇 面中标记为 id = ‘ seats ’ 的那些区域。 

当服务器返回 HTML 给浏览器时一切 正常。 但是现在它要返回 
JavaScript , 而我们并不希垡表单把它放到任何地方 • 我们希 望表羊 
能够执行它，这足非常不同的处理方式 • 

实际上我们需要在页面模板中做的修改很小。为了让表黾执行代码, 
我们要做的仅仅是去掉更新参数： 

:陳 

<% remote—forni 一 for (seat) do | f | %> 
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现在，当•个座位被 hut 时，表单会显示一个成功或者失败消息。 



I seats 


，心;广 


Right N*m« M«Q>0< 

a® 

-omcr^tw. 150 ^ {MliauttfiX 

19.0 S!SB 

luiien Co*'«nJ 1« 0 SfiflK fcflilQflUJa 

Cn*r. ie co»«in.l«0 Stlflat 跑 0*1 四 


“物 9 外。 


出珑 *5 磘认淡总 


Listing 


U^QfiakA 

U^OiaUfix 

CflilCcsxau 

iAiQuicax 

taiessuej 


New seat 


k 1 ^VL>£ 4 U 


New seat 


出现奋 刊表 t. 


但页面没有更新座位列表。我们需要生成额外 
的 JavaScript 来更新座位列表。 


^磨笔上阵 


你需要编写额外的调用来根据适当的局部橫板中的内容更新 

座位列表。 


page.replace 一 html 
: locals => { 


: partial 
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验证它 



B 完成的代码将实规几件 
搴惰 

已完成的代码如下 所示： 



我们可以随时调用 JavaScript 生成器 page 上的方法。所以 
如果座位保存正确， page 对象就会生成代码来更新通知, 
它也会创建 JavaScript 来更新座位列表。 
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ajax 




这个系统投入 r 运行， x 们能够快速地颅 vt 多个座位 
了。 
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rails 工具箱 





你的 Rails 工爯箱中的工異 

你已经把第7章收入驀中了，现在你已经把添 
jnAjax 到你的应用中的能力加入了你的工具箱。 
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H 样了 




你不可能一直取悦每 个人。 你能吗？我们 L ： 经肴过 r 你如 何使川 Rails 
来快速 H . 简电地开发能够完美满足一系列需求的 Web 应用。但迠如乂有 
其他需求出现的话该怎么办？如* :行作 人想耍基本的网页， 而 另外一鸣 
人则想要 Google 混搭 （ mashup ) ，还朽更多的人希喏你的;. V :用能够作为 
RSS 源 （ feed ) 存在，你乂该怎么做呢？你将在本取中创逑 MI 样基础数据 
的多种表现形式，让你能够以最少的代码来 七现最 大的灵活性。 


这是新的一華 




以用户为中心的设计 


在世界 各她登 山 

Head First Climbers 蛙为世:界各地的登山爱好者设计的网站。登山者 
在探险途中发冋报告，记录他们已经攀登的山眯的位置和时 W， 同 
时也报告他们发现的危险的特性，比如岩石滑坡和雪崩。 

很明昆，这咚信 息对 T 其他登山名的安全来说非常 策要， 很多登山 
者就迕接在岩壁 h 使用 T •机和 GPS 接收器來阅读和 kUf 信息。在这 
种使用方式下，系统将能够拯救很多生命而扎一一不管怎忭 1«1 
站也不会有太大的流量。 



那为什么它不受欢迕哝？ 

应用非常简单。它仅仅下面这个数据结构的支架版 


Incident 

山 fl|r (mountain) 

string 

纬度 (latitude r 

decimal 

经度 (longitude) 

decimal 

时间 (when) 

datetime 

标翅 (title) 

string 

描述 (description) - 

iext 


id 

mountain 

latitude 

longitude 

KZSBHim 



1 


63.04348055... 


2009-11-21 11:... 


tRubbie on the ... 



63.07805277... 


2009-1 卜 21 17: … 

Hidden crev^.. 

^Ice layer cove." 

i ： 

Mount Lotopaxo 



2005-0^-^7- T2:... 

Ascent 



^.Hiqh^ Kaaukl_irna 

^£392^_ 

72/7213^833^.. 

20M-0b~12J^^ 

Altitude si... 

- __ _ 



;遞龜•薩麵，支架是开均二个每_辞 方法‘ 轉需 
雜改通用6玄祕蘇！得更藏倉郎;的川 ) utumu 
解决的问题。 

那么这个应用需要改些计么呢?- 



创建一个 匹配这 个数据结构的 
支架式应用。 
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没庀多 X 时 Dll 便找到 M 站小受坎迎的原 W :用 户界面 

这个系统被 IH 干较理•々间数椐 _ Eid 泊 r 发生在令肚界的特定 

地点和时间的事件。位 W .13 息使用 F 而两个数据来 id 录: 


纬度 ( latitude ) . •这表征了地点在赤进以北或者以南的距离。 


经度 ( congitude ) „这表征 T 地点在本初 F 午线 以西 或者以东的距 


用户能够 n -: 常记请他 ff 1的 数椐: 他们 n ® 从 gps 掊收器里读取纬度 
和经度仿息就吋以 h m 他们作阅谈和解释来 fi pw 他镁山 荇的 
信息时却遇到 r 人麻烦。 


所以人 ( r ] 能够添加数据到《用中，似坫他们无法理解从这个应川屮 
获取的数倨。这减少 f 访 H 者. tfii 越少 的访问 荇意味荇越少的信 
息被添加……这将讀致史少的人来他用这个设用。这就是个恶性 M 


必须要做一些事了.否则网站就会失去更多的业务而不得不关闭。 


Wlt)^ 


请思考一下应用需要呈现的数据。你会如何显示这些信息呢？什么是最佳的 
手段来让这些信息能够容易被需要它的登山者们理解 •.》 





地图位置数据 


数椐 f 要在 ii ® 上 

系统 icl 录了地理数据，它应该被显示在地图上。 

正确的敉据能够被保存，而其他基本功能（创迖、读取、更新、修改 
和 刪除） 都可用。问題在干表达 ( presentation ) 。位置信息通过两 
个数倨来存 fift 纬度和经度-但这汴不意味肴我们必须以这种方 
式來显示它们。 

与其看到这样的内容…… 



不如让登山者看到如 F 的内容: 



显而易见，现在需要对界面做大刀阔斧的修改了，所以网站工程师决 
定他们不修改整个应用，而是首先运行一个小型的试点项目来创建 
一个能够显示事件并让它在地图 h 呈现的页面。但是他们不知道怎么 
做，他们需要你的帮助。 

你需要做的第一件事是什么？ 
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我们熏要釗建一个新的动作 

我们 不希® 修改现有的代码-一-我们只希望添加内容。在确保新界面 
可以丁_作之前，我们不希望让现有用户受到仟何影响。中竟，刺I、的 
已经不多 r …… 

所以我们将添加个名为 show _ with _ map 的新动作。现在， 柯人 
能够看到有个事件使用如 F 的 URL: 

http://localhost : 3000/incidents/l 


我们将创逑这个页 W 的新 版本： 

http: / /localhost :3000/ incidents/map/ 1 

这样，试点用户只需要添加 /map 来跳转到这个! Kifii 的新版本。我 
们使用如下的 路由： 


f ■二為= 


map.connect f incidents/map/ : id', : action=> 1 show_with_map *, : controller=> f incidents 


k 磨笔上阵 




我们通过拷贝 app/views/incidents/show, html.erb 文件来创 
建页面椟板。新文件叫什么名字呢？ 


事件 (incidents) 控制器需要一个新方法来读取合适的 Incident 模型对象并把它存储到名为@ 
incident 的实例变量中。请在下面写出新 方法： 
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测试你的祈动作 



s ihow 

3000/ ir 




我们通过拷贝 app / views/incident s / show , html . erb 文件来创建 

页面梭板。新文件叫什么名宇呢？ 
flpp/vicws/ii^cw^e^ts/sliow 


事件 ( incidents ) 控制器需要一个新方法来读取合适的 Incident 模型对象并把它存储到名为 
^ incident 的实例变里中。请在下面写出新 方法： 

. show wlth wtBp 吓 


现在创建页面模板和新的控制 
器方法。 


新动作 f 起来玎认工作 


+ 达样你/ 

t + 


现在，如果你卉看讲件沉商的两个版本，我们会朽•到他们都能够陡七 
正确的数据。你注意到 f 什么? 


show wlth .^vui-p 




飪 tfti 的两个版本 肴起宋完 仝相 M 这足-个问题。 


这一处 本夯赛 不同的 (3) 汾 

〆 


icnts show_with.map 

l&CilhoM J000Mncidtnt»/map/l 


ii 基琢象的 
支 3 甙负乇 


63.0434B05555556 


: 150.993963888889 


Latitude ： 63.0434805S55SS6 


Longitude: 150.993963888889 


Description: Rubble on the leoge tumbiea. 


Longitude 
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新页面 t 要一幅地樹……迖才是兵鍵 

我们当然不希項新版虹面看起来一样。 wn 需要添加幅地 m 。 

我们该怎么做？我们不可能搭违一个地阁系统。相反，我们将创达一 
个混搭 （ mashup ) 。混搭足指聚合了来 ft Web 中其他 M 站的数据和服务 
的 Web 应用。 


大多数地阁服务允许你在自 d 的 Web 应用驭嵌人地 m ， 而这儿我们将使 llj 
Google 提供的地 W 服务 。 Google Maps 給你提供 H 艮多灵活性。你小仅能 
够 ft 网 Iff 中嵌人地阁，你还能 个费太 多工夫就把你「1己的数据添加到地阁 
上，并 R 编程控制用户如何弓地 m 和数据交茳。 



K ifri 坫 这个豇 tfi i 如 M 丁.作的槪述 


会显水被 ii !^- 听怡的 人致位孜的地阉， HI 符 y •来 teitiM 确 f .>: 咒 - 


Head First Climbers 应用将生成调用地图的代码和鼴示在地图上的数据，俱 
坫地阁本讣以及 * 大堆允许用 P 执行类似 f •拖 动地阁 或荇缩放功能的代码 
将来 fi rGoogle Maps 服务器。虽然 Google 提供 了很多 代码，但我们依然需 
要提供两个 东西： 


调⑴地阁的 HTML 和 JavaScript 。 这有一些 U 杂，所以我们把所忠的 
HTML 和 JavaScript 放在不同的局部校板中，这样我们可以在面面模 
板中调用这些 W 部模板。 

我们需要在地 阁上显 示的数据。一开始我们将使用一个范例数据文 
件来确保地阁能够正常 n 作。 


那么地围代叭枒起来是什么柞了•呢？ 
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获取你的 g o o g k : 密钥 



我们需要卜‘面名为 _map.html.erb 的局部校板中的 代码: 


REpPq-4WZA270wgbtyR3VcV/ 


这儿没有足够的空间来显示所有 
的局部模板代码，但是你能够在 
http :// tinyurl . com/hfrailsmap 下载 
这个文件。 


这段代码实现 I ■什么?昏先，它调用了 Google Maps 眼务器 上的•性 
JavaScript , 这性 JavaScript 将在网页上生成-幅地图,该地图具有内嵌 
的拖动和缩放功能》 

但是堪本的 Google 代码尤法实现我们沿耍的一切。它汴不我人和! ul . 冶 
任何我们的本地 数据。 所以，在_ map . html . erb 局部模板中的代码 
还嬰从文件屮我人位 s 数据，它使 w 该数据来移动地 m 到 n : .确的位苦 
汴 a 指定点站不•个 
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迖段代码仪能工作在 本地主机 上 

Googleo 为 代码的使屮做 r 限制。他们强调你必须说明 
你将汴哪台卞机使用他们的代 W 。这就盘味 t 你在 w w w . 
yourowndomain.com 上使出免之前，你需要 古 •诉 Google 这- 
点。 为广 确 保人们 能够遵 守这个条款,这段代码仅在你提供 
给它一个 Google Maps 密钥时 t 能运行。密钥边 力一 个特定 
上机名而生成的，如果你 Ut 图把 Google 地阁 嵌人到 丼 他地方 
的页面，地图将拒绝运行。 

但这对我们而言不是 fr 4 SS 。我们使用 fi * J _ map . html.erb 
局部梭板包含 f 针对 本地 + i 机的 Google Maps 密钥 所以 
只要你在自已的机器上运行就没有 问题。 不过请记住，当你 
作其他地方运行这段代叭时，你怎要宄屮沾你 Ad 的密 
钥。 


扪穹魟钕— 

如果你想要在自己的 Web 应用中嵌入 
Google 地图，你蕪要与 Google 签署 
协议。为了做到这 一点， 请访问如下 
URL ： hUp :// tinyurl . com / mapreg 。 




笔上跸 


你需要在 show _ with _ map.html.erb 模板中包含局部模板 map 。 我 
们需要传入一个胃名为 CU & 的具有地图数据文件路径的本地变 M s 我们 
将使用测试文件 /test.xml 来实 现这一 目的。 

锖编写调用局部模板的代码。 
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发送数据 Mgoogle 




笔上晬 
解答 




你需要在 show _ with _ map.html.erb 模板中包含局部模板 map 。 我们 
需要传入一个名为 data 的具有地图数据文件路径的本地变量。我们将使 
用测试文件 /test.xml 来实现这一目的。 

请编写调用局部模板的代码。 

Ytv^dtr (: pflrtl«l=> 'Kvurp' ,: Locals =>{: data = > }) °^> 


现在我们 f 要 M 穆数椐 

在我们能够验证嵌人式地阉之前，我们需要为它提供地 m 
数据。作为开始我们将直接使用 test.xml 测试文件它的内 
容如下所示 : 



public 


为了免去你输入这么多数字，你 
可以从 http://tinyurl.com/maptest 下 
载 test.xml 文件。 


test.xml 

_ 

is an example de 似讲 ㈣ 

<Utit U de>63.0434805555556 </Xatitude> 

<lon g itude>-150. 993963888889</lon g itude> 

<title>Test Data</title> 

</data> 


地 m 数据提供 r 测甙 • 例 1 的邰度和经度。当我人 Google 地 
m 时，我们 的局 部模板 map 将把这个文件的内咨拎递给它 , 
这样 事汁就 能够被居中显示丫。 
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式劈—— 

那么当我们输人如 FURL 会发生什 么呢： 
http://localhost:3000/incidents/map/l 



地图工作了！但如果我们想跳转到•个不同的 URL 呢？ 



无论选择什么样的亊怍，毎幅 地图荇 起来都-样。这足因为毎幅地阁都在 
使用相同的数据： test . xml 文件的内容。 

为了让地图显示指定事件的位置，我们需要为每个页面生成不同的数据 
文件。 
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生成你的 xml 


我们 f 要生成什么样的数椐哝？ 

我们把 XML 数据传递给地图，而 XML 数据描述了单个事件的位罝。 
每个位置包括纬度、经度、标题和描述。我们需要为每个事件生成这 
样的 XML 数据。 

所以系统将如下 工作： 



O 


1 ^) '-h 

笏搴 4 的 XML 款旖 





(/incident 》 




豫务器'主咸攀碑的 ML 


如果你感到似曾相识的话就太好了！ Google Maps 实际 t 使用 Ajax 来 
运作。还记得我们在前一章如何使用 Aja X 来下载座位列表的新版本 
吗？ Google Maps 使用相同的方法来为每个亊件的位置请求 XML 数据。 


所以下一步就是生成数据。我们从哪儿获取数据呢？ 
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我们从糢型中生成 XMl ]- 

* 成的 XML 数辦将来自于 Inc i dent 模型。我们将只使用四个 域性 : 
纤度 (latitude) 、 经度 Gongitudqh 标题 (title) 和描述 (descriptipiv }。 


XML 和多种表现形式 


fK 我们如何生成 XML 呢？在某种程度上，这有点悚生成 NlS。 毕 
-&XML 和 HTMLlh 常 W 近。两耳：电如网环运舍 iUl j 校 M 的数 
据那样，我 r 啲 XML 女作也從每含来_自于模吧的数据。 

所以二沖方法就是创违一个包含 XMLfe 签 rfii 不是 HTMLfe 签的页 

面 模板： ，:.，:， 


Incident 

mountain 

s,t 1 ing 

latitude 

decimal 

longitude 

decimal 

when 

. 7 .； 

title 

string - 

description 

text^- 


app 


m 

L> views 

L> [jjttrf incidents 

I 笔 show_wi^h_map.html.erb 


.^ 9999 .—_ 55 _ 4 - 个辑 ㈣ m a 板: 

^ 11 ) "" (S 我们在技这么漱嗜二 


这种方法能够奏效，但是我们有更好的方法…… 
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让你的模型生成 kml 

模型对象能够生成 XML 



模型对象包含数据。 XML 文件也毡含数据 。•所 以龙种意义 
.上模型对象就能生成它们的 XML 版本。_.毎个模型对 象有: 二 
个返回 XML 字符串的方 1法 to xml： 


这 4 辜智的象 - 


@ incident.to xml 

：,： T 一 

toJC 从 l ; 法达® 3 一个代表这 
个俅嗲 的象的 XML 穹符赛 



m 是创建 XMLDU 乂完成 r 一半的[:作。另一半是把 XML 
返冋给浏览器。我们没有角页面模板，所以所有的工作不 
得不交给控制器来处理渲染 XML …… 
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控制器代码看起采是什么#嚯? 


我们 "r 以修改 show _ with _ map 方法来输出 XML : 


这4 戧 ( HSS 4 该 K 的 
时篆。 

rt^dtris ® 
XMU 0 


def show_with_map 

''~~^ @ incident = Incident. find (pa rams [: id]) 

-> render : text=>@incident. to xml < — it _ 个猫这的象的 
end ' ? _ 緣柳。 

洽别 US 杓垃戏. 


render 方法把 XML 返 In ] 给浏览器。我 们前面 见过这个 render 方 
法，但是这次稍稍有些区别。大多数时候你使用 irendei : 来从一个 
揆板或者 W 部揆板中生成 个 网页。 m 是你也以只给它传递一个 
字符串对象-而这就是我们现在的做法。 


扪穹魟妫- 

为了简化编程， Rails 工程师也允许你给 
render 方法传递一个名为: xml 的参数： 

render : xml=>@incident 

如果 render 方法被传入一个使用 : xml 

参数的对象，它将调用对象的 to _ xml 

方法并把结果返回给浏览器。 render 

命令的: xml 版本将生成与我们的控制器 

里的 render 命令一样的内容，但是它 

也会把响应中的 mime - type 设置成 text / 

xml 。 不过现在我们还是用上 面的 : text 

版本。 

♦ 



问 


没存蠢河题- 

有 蠢河题 


再说一遍，这次 render 方法做了什么？ 


• render 为測見器生成一个响应。当你的浏見 
器要求一个页面时， 那就是请求。 render 生成需要返 
回的内容。 
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试驾 



控制器现在返回 rXML , 它包含的数据来自干 id 
象。 


1的事件对 


但有没有问题呢？我们生成的 XML 看起来和范例 XML —样，但 
还是有一些 不同： 


我们生成了太多的厲性。例数据文件仅仅包含 
包括纬度、经度、 标题 和描述信息。但是这段 
XML 包含了事情的所有内容，甚至包括了事情 
记录被创建的日期和时间。 


XML 文件的根名是错误的。生成的 XML 从我 
们使用 的变量 屮获取 f 根名， 〈 incident 〉。 
但是我们需要一个根名为 < data ：^ XML 。 


<data> 

<de S cription>This is an example 

description</description> 

<latitude>63.04348055S5556 </latitude> 

<longitude>-150.993963888889</longitude> 

I <title>Test Data</title> 

I </data> _ 


XML 的格式基本正确，但是还有一点点瑕疵。 


我们需要修改 to xml 产生的 XML 。 




那么现在当我们到达如下 URL 时会有什么结 果呢: 


http :// localhost :3000/ incidents / map/l 


eoo 


< ?xml vcrsi ort^ 11 1,0 

<incidont> 

<croated>at typo 


encoding* 


<croated-at typ«datetime*'>2009-11-21T11 s59s 31Z</created-at> 

<descriptionRubble on the ledge tumbled, and just missed aa.</doscription> 
<id type* •” integer">l</id> 

<latitudo typ«*"decimal M >63.0434805555556</latitude> 

<longilude type*"decimal">-150.993963888889</longitude> 

<mountain>Mount Rushle8S</mountaio> 

<title>Rock alide</title> 

<updated-at type-'datetirae">2009-ll-21Tll:59:312</updatod 
<when typ««"datetimo">2009-ll-21Tll*55s00Z</whon> 

</incideat> 
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0 


to _ xml 方法 有一些 可选的参数能够让我们修改返回的 XML 。 
看 i 你是否能够找出这些参数应该被賦予 的值： 


def show_with_map 

@incident = Incident.find(params 【： id]) 
render : text=>@incident.to 一 xml( 

. =>[ 

=> ) 

end 


G3 


|^change^k 


: title 


j^«scriptionk 


: longitude 






^onlyj 


j^data J 


: root 




j^exceptj 
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to _ xml 方法有一些可选的参数能够让我们修改返回的 XML 。 
看 i 你是否能够找出这些参数应该被賦予的值 


©.於我 f)_ 值用） 
用令制 


我们句 W •(吏 


済蠢河 


你可以这么做，但这并不是 个好主 意。你可能需要在不同的场合生成不同的 XMG 如 
果你在樸型中为每祌 XML 格式添加代码，模型将很快就会过栽。一 - : : 
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现在，当我们来到如下 URL 时： 
http://localhost:3000/incidents/map/l 

我们将得到一个有些变化的 XML 。 


科 。 O _ Source of: httpj/io^al^^tiBOOO/ incidents/map/1 _^ 

<?xml versions"1.0^ encodings n UTF-8 n ?> 

<data> 

<description>Rubble on the ledge tumbled, and just missed us .</descriptioD> 
<latitude typ«» "decimal ">63.0434805555556</latitud©> 

<longitude typ«»"decimal">-150.993963888889</longitudo> 

<titl®>Rock slide</titl«> 

</data> 



你已经能够修改 XML 来使得它仅仅显.示我们需要的数据，而且它 
也有了一个恰当命名的根元紊。现在它看起来比较接近原来的范例 
XML 文 件了。 

to _ xml 方法并不允许你对它所生成的 XML 做大量的修改，但是它 
对千大多数应用来说已经足够……也包括为定制地图而给 Google 发送 
XML 的情况。 

通过很小的修改 ， to xml 便给了我们所霑要的 XML 。 
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.... .- 

^ 有些使用试点程序的人遇到了 问题。 

f\ •次攸改之 m , 如卜•的 URL : 

' http :// lbcalhost : 3000 /in cide nfes / map/l 


能够生戍网 ljU H 题足,现江这个 URL 仪仪返 MXMU ^ 

亮的 Google 地图。 ， 


在你的最后一次修改 之前: 


此时，在20 000英尺的窩度… ••• 


-糾汝二*.. 残加-个极找 吵 
嘁 ® iS 子我们教旖的用费 


在你的最后一次妗改之后 

rt on 

.-:.v-d、:....^a vcr 5i3n -*1.0 


JSl ； :的⑽ 5 

I 


Source of: http://locaihost:3000/incidents/map/1 


<?xml . ver jx6i»*»' Z . 0 *■. encqdi ng>< ".UTF -： B ^ ^ 

<data>_ 

<doscription>Rubble on the ledge tumbled, and just missed as .</doscript: 
<latitude type 'declir.a ： >63 70434805555556</l«titudo >-， 

Vlongitude typ«- , MecirMl ,, >-150.993963888B89</longit~udo> 

<titlo>Rock alide</titlo> _ 

</data> - :' ' ■■ • ' ' -V 


登山者也需 要网站 
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show _ with _ ma ^> 动作原来会使用 show _ with _ - map . html > 
erb Iff 面模板生成一 f 网页。但是一旦我们为控制器方法添加了一 
个 render 调用之后， Rails 就忽略了那个模板而仅仅生成 XML : 


当然，这个逻辑是成立的，因为动作不可能同时生成 XML 和 
HTML 。 、 


但我们还是需要一个网页来显承地图，而地图也依然需要 XML 地 
® 数据。我们该怎么办呢？ 
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Mark : 另一个动作? 

Bob : 当然，一个用来生成 XML 而另 -- 个用来生成 HfML 。 
Laura : 奸吧， k 不是 * 个好主 k 。 

Bob : ^什 么? 

Laura : 这意味着 敷复的 代叭。两个方法都需要编 W 代码来读取 
事件对象。 

Bob : 那有什么关系。就一行 rfri 已。 

- - . . •• • , . - • •. 

Laura : 目前来说是的。似是如采我们将来迕要修改•些地打呢？ 
Mark : 你是说:比如模型被修改？ 

Laura : 或者也可能是们要从其他地与获取数据的情况，比如 
从一个 Web 服务。 

Bob : 这没什么大不了的。 让我们 先考虑我们现在的问题吧，好 






Mark : 我不知道。 Laura , 你会怎么做呢? 

Laura : 很疏单。我将给动作传递个参数，告诉它我们偌要什 
么样的格式。 

Mark : 这应该能行。 

Bob : 得了吧，太抡工夫了。 

Laura : 比创建另一个动作花的代价要小。 

Mark : 值还有一个问题…… 




Mark : URL 不是确定丫我们需要的信息吗？ 

Laura : 所以?- --..' ; 

Mark : 我 H 不是要为这两种格式使用相 R 的 URL 吗？ 
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XML 和 HTMI • 仪仪 R 是表规形式芮 B 


虽然 HTML 和 XML# 起来大不样，它们其实 H 是相 N 内容的不同视 
觉表现形式而已。 HTML 网页和 XML 地图数据都描述了相同的 Incident 
对象数据。事件是核心数据，有时它被你为资源。 

资源足网页所呈现的数据，而网 K 则被称为资源的一种表现形式 
(representation) 。 以 Incident 对象为例。 Incident 对象就足资源，事件 
m 豇和地阁数据 XML 文件都是资源的表现形式。 


逐资满 





把 Web 想象成一系列资源和表现形式蛙名为 REST 的设计架构的组成 
部分。 REST 也是 Rails 的架构。你的应用有越多的 REST 成分，它会在 
Rails 上运行得更好。 


但这对我们有什么帮助呢？好吧，为了史好地遵守 REST，XML 数据和 
网页需要使用相同的 URL (Uniform Resource Locator, 统一资源定 
位器），因为它们都代表了相同的资源。 比如： 

http://localhost : 3000/incidents/maps/l 


但为了 U: 事情变得简单，我们可以 (稍稍) 妥协一下 REST 设计而 
使用下面的 URL 来实现两种丧现 形式： 


http://localhost : 3000/incidents/maps/l. 
http://localhost:3000/incidents/maps/l. 


xml 

html K 


一个 Ci ® HTML 
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选择你自己的格式 

我们应泫如何碥定使用哪一种掊式? 

如果我们添加-条路 径中包 含格式的额外 路由： 


map. connect • inc i dents /map /: id.: format', .:action=> f show 一 with 一 map •, 
: controller- 〉 *incidents* 个 一 - 、 

护幕名象 

我们将能够从 XML 中读取所请求的格式，然后再根据如下的代码来 t 
作 决定： 


if params [ : format ] 


， html ， 


Generate the HTML representation 


else 


Generate the XML representation 


end 


e 




( i 个护 f 名枵破保乓 
在: format 域中 




但大多数 Rails 应用并不使用这种方法来确定生成格式。相反它们调用 
一 个名为 respond _ to do 的方法和一 个名为 responder 的对象： 


respond_ to do | format | 




format.html { 


} 





format • xml { 

___ 一*这凡基坌威 XAiL 的代蓚 

> 

end 

这段代珥实现了相似昀功能。这儿的 for mat 对象是个响 7.5 器 
( responder ) 0 响应器能够根据请求所要求的格式来确定每®索要 
运行代码。所以当用户请求 HTML 时，以上代码就会运行传递给 
forma . t.html 的代奶。当用户请求 XML 时，晌应器会运行传递给 
format 的 代码。 

1什么 Rails 程序员不直接使用 if 语句呢？毕毚，这样不是史简单吗？ 
好吧，响应器具杆额外的能力。例如，它可以设晋响应的媒体类型 
(mime type ) 。 媒体* 喂告 诉浏览器响应中包含了怎样的数据类型。 
通常来说，推荐使用 respond _ to d 6 来_定生成明卩一沖丧示格式。 
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控制器中的 shpw _ with 」 map 方法 需寒确定它该生成 XML 还是 HTML 。 诮编写这个 
方法的妍版本来使^响应器生成正确的奉现形式;。 

提示\如果你需要生成 HTML ， 除了读取模型对象外，控制器还需要做什么？ 


show _ with ^ map . h £ ml . erb 页面模板当前调用 map 局部模板并把 / tes ^. xml 文作传 递给它。.如裝该 
页面模板要调用生成的 XMI ^ 文件，局部模板凋用应该是什么样子呢? 




respond_to do 的正确实现 





控制器中的 show _ with _ map 方法需要确定它该生成 XML 还是 HTML 。 诮编写这个 
方法的新版本来使用响应器生成正确的表现形式。 

提示： 如果你需要生成 HTML ， 除了读取模型对象外，控制器还需要做什么？ 


热爸^ 提示： 如果你需要生成 HTML ， 除了读取模型对 f 

show_wltV< 从 fl?dcf show_with_»H«p 
ju 

保取 3 (^iy^ude^t = (parawcs[ ： Ut]) 


rcsppw-c|J:p do J format | 


:把^乂 ……… • iiJtgvM ? 一 


rt^tr : text = >@l^cldevit.to jcwil( 


： o»Uij=> [: Lfltttu.dc ,： U>kv0ttM.c|lc, : ：olesc 


show _ with _ map.html.erb 页面横板当前调用 map 局部栈板并把 /test.xml 文件传递给它。如果该 
页面模板要调用生成的 XML 文件，局部模板调用应该是什么样子呢？ 

<^ = = ?. ! )^1?• • .，• ：^«Ls == ? . : 牙二 .).)• .?• ？. 
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当我们査看位于以下 URL 的 XMt 版本的页 面时： 
http://localhost:3000/incidents/map/l.xml 
我们将得到事件的 XML 版本： 


^ 0 P 


Source of; http://localhost: 3000/ incid«nts/map/lj 

oQcodin.g-' 'VTF-8 x ?> 


<do»cription>Rabble on the ledge tumbled, and just missc 
<latitudo typo• N dociinal">63.0434805555556</latitudc> 
<longitudo typ«^ ■_decimal**>-150.993963888BB9</longitudo> 
<t£tlo>Rock slide</titlo> 


.</deacription> 




那么 HTML 版 本呢： 

http :/ 7 U>ca Lhost 八 cicteKvts/Kvtop/i. 

' n ' " 一 - 
-i I Cji.'i i ♦ »*•>**'• »：« - «J* I ■»** 






■ 

mi»Ma» CMMn 
I 』 ■« «»ne«vMMVi • 

W*mm： ^w a' ll-M . 

sat ■: — .. _ 



系统工作了。现在不间的举件显示了不向的地罔。何在我们替换当 
前运行版本之前，我们最好确保.我们真的理解代码是如何运作的 。 

这究竟是怎样发生的呢？ 
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浏 KT 器指向了 HTML 版的页 :面^ 拉制器 意识到 i 対览器济求的足 HTML 而不是 



JavaScript 请求 Google 地图。 

N 豇屮的 JavaScript 向 Google Maps 服务器沾求地阁数据。 Google 服务器返 
回该数椐。 










JavaScript 请求事件的 XML 数据。 

.N0I 中的 JavaScript 从控鈿器那‘儿请求事件的 XML 数据然后七在' 
地甩上®示该数据。. 
















— X - - ---:—— 

々存蠢网薇 

请求中包含 “ Accepts :” 这-样的头来式通常对 Ajax 请求返回 “ true ” 而对 
良明 比如 请求 “ texl / xml ” ，同单的 洌見 器请木笔回 “ false ” 。问 
邶么响应器将运行代码来返回 XML 格题在于它只造用于 Prototype 4，它并 
人。 不适用于所有的 Ajax 库。 


有没有什么方法把你不希望丨为什么有时 
包含在 to _ xml 输出中的属性列举出用 render 而有时又不用 




& • :•.如 果你悉 意.负用默认撗垮 ( /f 
令：有 ( 我奵寸以不使用 : bhly 丰与动作的馍 &) 6 就兮以 ^ 

参致， H , 我 ffV 使用 ： e xqep t 参略 [ ei > de ^ 调 . 

致 Raih < i 二致性方面很显著，你 . ^ 

+发现 Rails 中有若干地方具有可选参 Pj #你说过被生成的 XML 和 HTML 
致: prrly t - 在所育 ii 4 地方你却 T 以只是不同的表现形式，但差它彳门并没 
b 换到： ej < c _ ept 参数来表明呷斧冬你有包含相同的信息，对吧？ 

木需要的。. : - 

答： '确实 • 它们的信息并不相 

有方法可以让控制器判断出同。为单一亨件生成的 XML 包含了比 
这是来自于 JavaScript 的 Ajax 请求还 HTML 表现形式束少的数据 e 但是它 
是籴自于浏览器的请求吗?. 、 们都表达了同一资派的信息，所以-它 

奵都是相同事件的表现形式 6 

• 算有吧 9 request .xhr ? 表达 


是的..当然是这^ .把格式加 
入到 URL 中—是 史 iRgST 式设计 7 ^ 一 
:种拆 1 中做 法. 二…_至’少有一 4:点是 it 蛘. 
长:它是一种 :很常見妁:4巧 e 它很 f " V 单：. 
而且运作#很好 ： • ' — 


■ 


有一种方法可以实现！.如茉 






让一些登山者试试吧 

r ■-七 • _ . ■■ . . - . - . ... . . ^ -W ... : ■ 

代码可认正式 运行了 




我们的新版位璧页面工作得挺好，所以让我们用 show with _ map 
代們来_换支架式 “ show ” 动作吧。: 护思贤 






删除路由。 

我们为测试代码创 建了自 定义的路由；所以我 
们需要从 routes.rb 文件中删除 它们： 


在控制器中重命名 show__with_map 方法 。_ 

s h o w _ w i t h _ m a p 将会成为我们的新 s h o w 
方法。.所以让我们刪除现钉的 s h o v ; 力•法并把 
show _ with _ map 重命名为 show 。 


然后重命名 show 一 with 一 map.html.erb 模板。 

这是说我们需要删除现 W 的 sh 0 w . htmi 4 : rb 并 
把它替换为 shgw with map.htrni . ert > 模 


它生成支架要用的标准路由 


• map . resource 路由设 I 了 完 
整的 路由。 这巷路由也包括格式。 


如果没有指定格式， RaHs 会 
伋定它是 HTML ……这对我们来说 
也比较方便。 
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■ 


现在地图版本的页面已经替换了默认的 “ Show ” 动作。 
所以主索引页面链接到了地阁页面，而不是文本页面。 


Listing incidents 


匕二 M 0MMJ777T77B 



























改进索引页面 





用户已经开始询问索引页面能否显示所有被记录下来的事件，而幸运的是，只要我们 
提供正确的 XML 数据，_ map . html.erb 局部模板就能够生成多点地图。 


下面是事件 ( inciden ⑻控制器中现有的 index 方法。请重写这个方法来从所有事件的数 
组中生成 XML 。 你只需要将根元素修改为 “ data ” 就可以了。 


def index 

@incidents ■ Incident.find( : all) 


respond 一 to do |format I 

format.html 參 index.html.erb 
format.xml { render : xml ■> @incidents } 
end 
end 



trailers 

incidents_controller.rb 
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.索引页面需要包含一幅地图。编写代码在指定点插入地图。你需要把 XML 版索引页面 
的路径作为地图数据传入。 

<hl>Listing incidents</hl> 

<tafcle> - - 

... .. •••—.. •• • • V. * •- • •_ \ 

<tr> . 

<th>Mountain</th>.. ~ 

、 <th>Latitude</th> ... 

- <th>Lon^itude</th >. ： 

< th>When< / th> 

<th>Title</th> : 

- <th>Description</th> - . 

<% for incident in.@incidents %> —— 

<tr> 

<tdx%«h incideiit.mountain %></td> 

<td><%=h incident. latitude %></td> 

<td><% 腫 h incident .longitude %></td> •_ 、 - :: 

<td><%-h incident .when %></tdi> 

<tdx%»h incident .title %></td> 

<tdx%=h incident.description %></td> . 

<td><%= iink__to 'Show*^., incident %></td>. 

<td><%= link_to 'Edit*, edit—incident—path(incident) %></td> 
<td><%» link_to 'Destroy', incident, : confirm => *Axe you sure?/, 
;method => : delete %></td> 

: </tr> •； . ... 

<% end ' -. - 

彳 /table〉_ .. 


• ^ • V ； ： '. v - '• ... . . ... • • • • ■ ... • 

<%= : link_to 'New incident ', new_incident_path %> 


卿 

Us 



incidents 


index.html.erb 
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地图化你的数组 





用户已经开始洵问索引页面能否显示所有被 id 录下来的事件，而幸运的是，只要我 
们提供正确的 XML 数据，_ map . html . erb 局部模板就能够生成多点地图。 


F 面足事件控制器中现有的 index 方法。 W 班写这个方法来从所飪事件的数组中生成 
XML 。 你只需要将根 7 d 素修改为 “ data ” 就吋以了。 


def index 

@ incidents = Incident.find( : all) 


respond_to do I format I 

format.html # index.html.erb 
format.xml { render : xml => @incidents } 
end 
end 


(: flU.) 


rts^o^d_to do |forwu»t| 





rtv^dtr : Uxt = >@ lv ^ cidtv^.to jc » aU (: root = > " dfltfl ") 

} 


t^d 
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XML 和多种表现形式 


索引页面需要包含一幅地图。编写代码在指定点插入地图。你需要把 XML 版索引页面 
的路径作为地图数据传入。 

<hl>Listing incidents</hl> 

<table> 

<tr> 

<th>Mountain</th> 

<th>Latitude</th> 

<th>Longitude</th> 

<th>When</th> 

<th>Title</th> 

<th>Description</th> 、 

</tr> 

<% for incident in ^incidents %> 

<tr> 

<td><%=h incident.mountain %></td> 

<tdx%=h incident. latitude %></ td> 

<td><%=h incident.longitude %></td> 

<td><%=h incident.when %></td> 

<td><%=h incident.title %></td> 

<tdx%=h incident.description %></td> 

<td><% ss link_to ' Show', incident %></td> 

<td><%** link—to 1 Edit ', edit_incident_path (incident) %></td> 
<td><%« link 一 to 'Destroy', incident, : confirm -> 'Are you sure ?*, 
: method ■> : delete %></td> 

</tr> 

<% end %> 

</table> 


<f= rtv^dtr (rpflrtlflt= > Wp', ： lo&flU=>{ ： o(fltfl = > 多 > 

<br /> 

<%- link—to 'New incident*, new_incident_path %> 
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.. : . •— •； ,~r . * . ... -r- , • 

二个宪全图形化的索引贡 面，… :": ■ 

A 劈 — 

琰 ft 当用户*到首页时， te 们吋以看 到地图 h 的事 件列表 ; g 
革个-事件被点击时，详细信息会被 k 示出来，同时还会显示二个 
抬向事件自己的贡面的链鯈。 - 
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XML 和多种表现形式 




大多数网站现在都提供 TRSS 新闻源 (news feed ) 
够提供指向某个网站上的主要资源的便捷链接。 


它能 


但一个 RSS 新闻源是什么样子呢？ 
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rss 就是 xml 


RSS 源就是 XML 

登山 M 站的 RSS 源文件如下 所示: 


<rss version*"2.0"> 

<channel> 

<title>Head First Climbers News</title> 

<link>http://localhost:3000/incidents/</link> 

<item> 

«I— " a. .s, .3^ U s, /deS c tiP Uo n > 


<Unk>http://loc a lhost:3000/incidents/l</nnk> 


</item> 

<item> 


这就是一个 XML 文件。如果你使用 RSS 新闻阅读器，或荇你的浏览器 
能够 U ■阅 RSS 新闻源，它们会下栽一个类似的文件，这个文件将包含 
指向新闻故亊的一组链接和描述。 

那么我们如何生成一个这样的 RSS 源呢？ 


^ ^獅姑 - 

RSS 中有没有什么标签让你看起来觉得特别奇怪或者不清楚吗？你觉得 
channel 用来做什么？ link 呢？ 
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XML 和多种表现形式 


我们将釗建一个名叫 news 的动作 

让我们这样创建一个新的 路由： 

map.connect '/ incidents/news * f : action =>' news ', : controller *〉’ incidents ’， : format=> , xml 


、磨笔上阵 


请编写出这个新动作的控制器方法。它需要通过 updated _ at 
来找出过去 24' 小时内的所有事件。然后它应该通过调用匹配事 
件的数组的 to _ xml 来渲染默认的 XML 。 

提示： Ruby 表达式 Time . now . yesterday 返回一个正好是24小 
时之前的 date-time 值。 
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渲染你的 rss 
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XML 和多种表现形式 



这是 news 动作生成的 XML: 


it(i 一鸯的两 
榷奚 的， M 以 ‘ 0 •奄 
一押赛24.).妁〔内 ? 皮 
外拴过 护辜碑分含芝 
条出來。 


我们为正确的数据生成了 XML， 但是它的格式并不是我们的 RSS 新闻源所期望 
的。这就足够好了，我们以前遇到过柑 M 的问题。当我们为 位置数 据生成 XML 
时，它也是错误的格式，而我们能够纠 正它。 


把这个 XML 转化成能够匹配 RSS 新闻源的数据结构有没有什么问题呢？ 
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控制你的 xml 


我们必领更改 XML 的结构 


to _ xml 方法允许我们对它所产生的 XML 做一些 简黾的修改。我们 
可以交换名字以及选择需要包含的数 据项。 但它是否足以让我 ( N 把现 
有的 XML 变成我们想要的 XML 呢？ 




户 "1.0” encoding^^UTF-S" ?> 

《incidents type*"array "〉 

〈 incident 〉 .. oi T ii•59• 31Z</created-at> 

rr us - </description> 

<lo ； g nu d e ty pe=Mec im ar.>-lS0.99396388B8B 9 </lon git u d e> 

< m ountain>Mo U nt Ru S hless</mount al n> 

<title>Rock slide</title>_ __ " 


<rss version= H 2.0 M > 

<channel> 

<title>Head First Climbers News</title> 

<link>http://Xocalhost:3000/incidents/</link> 

<item> 

<title>Rock slide</title> . 

<description>Rubble on the ledge tumbled, and just missed us^/descrxpt.on 

<Unk>http://localhost:3000/incident S /l</link> 

</item> 

<item> 


我们 t 要更强犬的 XMl 处遑能力 

新闻漉 XML 无法通过 to _ xml 方法来生成。 M 然 to _ xml 能够轻微地 
修改 XML 输出，它无法沏底改变 XML 的结构。比如， to _ xml 不能在 
不同层次间移动元素；它也不能把多个元素组合 起来。 to: xml 被设计 
用来快速 a 简黾地调整 XML 输出，但这也导致了它缺乏灵活性。 

为了获得更强大的 XML 处理能力，我们 需要一 些其他的技术…… 


•fsii 基戏们 S * 杓 
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XML 和多种表现形式 


所 认我们 将使用一种新 模板 : XMl Wilder 

如果我们创途-个 HTML 页面模板，我们就能够生成任 H 我们想要的 XML 。 毕 
竞， HTML 和 XML 非常 接近： 


<rss version="2.0"> 

<channel> 

<title>Head First Climbers News</title> 

<link>http ： //loc a lhost:3000/incidents/</link> 

<% for incident in @incidents %> 

<item> 

<title><%» h incident.title %></title> 
h incident.descriptio 




<description><% s= 


%></description> 


不过 Rails 提供 r 一种特殊的校板类型，它特别设计用来牛.成 XML , 你 
为 XML Builder 模板 （XML Builder Template ) 。 

XML Builder 与页面梭板在相同的0录下，而 K 它们的用法也相似。如 
果 也人请 求一个 XML 响应（通过在 URL 未尾添加 . xml ) ，控制器只需 
要从模塑中读取数据， Rails 就会自动调用 XML Builder 模板。这就息味 
宥我们可以从 news 动作中去掉•行 代码： 




def news 

(^incidents = Incident. find (rail, : conditions=> [' updated_at' / > ?•, Time .now. yesterday]) 

™^ I?inci dentd* 1 

end 


现在这段代叭将直接从模喂中读取数椐，剩 T 的活都可以交给 XML 
Builder 换板了。 

那么 XML Builder 长什么样呢？ 


app 



incidents 


苟爷榉板'主 
y(T 

show.html.erb 


XML ^Udtrm 


news.xml.builder 
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xml builder 


沐六 NJVi 厶 Builder 


Iftllf 模板设计得就像撒了些 Ruby 代码的 HTML 文件 。 XML Builder 
则不同。它们是纯粹的 Ruby ， 但是它们会带有一个类似于 XML 的 
结构。例如下面这个： 


xml.sentence (: language=>'English') { 
for word in @words do 
xml.word(word) 
end 


将生成如下的 东西: 


〈sentence language= M English w >^ —綦性 
<word>XML</word> 
<word>Builders</word> 
<word>Kick</word> ^ * 

<word>Ass!</word> 

</sentence> 


那么为什么 RailsI 程师引入了一个不 间的模 板呢？为什么 XML 
Builder 不像页面模板那样工作呢？为什么它不使用嵌人式 Ruby 呢？ 
即使 XML 和 HTML 非常接近——在 XHTML 的情况中它们从技术上 
宋说是一样的-人们使用 HTML 和 XML 的方式仍有些不 N 。 


# 网页通常包含了一大推 HTML 标记 来让页 面变得漂亮，而來 A 数据 
库的数据只占网页的很少一部分。 

# 另一方面， XML 的大多数内容更可能是来自于数据和条件逻辑，它 
极少可能来自于 XML 标记。 

使用 Ruby ——而不是 XML ——作为主要的语言，使得 XML Builder 
更褚简也更易于维护。 
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使用 xml 模板 


代碚泜键拯斛著 




你的任务是从代码池中选出代码片段， 
并把它们放到下面代码中的空白 
_处。每个代码片段只能使用 一次， 
fk 而且你不需要使用所有的代码片 
/段。你的目标是完成生成 RSS 的 
XML Builder 模板。 


|1^| app 
U jjjjp vie ， 


incidents 


. I ： version=>"2^1 

xml. 


U 1 


news.xml.builder 


xml.title( 


•Head First Climbers News” 



xml. link ("http: //localhost: 3000/incidents/'*) 
for incident in . 0 incidentaj|. 


xml.it€ 


title 


[incident.title) 


.description 


incident.description 


承：: 


> _ 
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现在让我们拕源加入到 克靣中 

但用户如 H 找到这个 源呢？ 浏览器通过査找页面中的 < i ink ... />引用 
来发现新 W 源的存在。 

Head First Climbers 的工作人员希望新闻源能够出现在毎个页面上，所以 
我 fH 通过 auto __ discovery _ link 辅助函数把指向 RSS 源的引用添 
加到 incidents 布句文 件中： 


<! DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN w 

"http: "www.w3.org/TR/xhtmll/DTD/xhtmll-transitional.dtd"> 

<html xmlns=*'http://www.w3.org/1999/xhtml M xml : lang= M en" lang="en M > 
<head> 

<meta http-equiv= H content-type" content®" text/html; char set=UTF-8*' /> 
.<title>Incidents : <%■ controller.action_name %></title> 

<%= stylesheet_link_tag 'scaffold' %> 

<%= auto_discovery_link_tag( : rss, { : action=>'news'}) %> 

</head> 

<body> • 

<p style='*color: green"><%= flash [ : notice) %></p> 

<%= yield %> 

</body> 

</html> 


这将创建如下的 链接： 

〈link href-”http : //localhost:3000/incidents/news.xml" 

rel="alternate" title="RSS" type- t 'application/rss+xml» /> 

但为了验证它是否工作，我们需要再次启动我们的 
Web 浏览器。 


|fl|^| app 

views 

L>r~i |a v 


outs 

incidents.html.erb 
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谁来试试 rss? 



现？ t ,' ■当用户来到 M 站，一个 RSSi ) S ( ttlfc 就出现在他们 

的浏览 器中： 琛？-个* 甬; 















XML 和多种表现形式 

在世界最窩点 f 

网站的第-手新闻中存•条是我们无搜的登山者贴出的，数以千 
计的登山者听到了这个好消息。 






_ 


rails 工具箱 


oo 


爹 


你的 Rails 工異箱中的工與 


/ 你己经把第8章收入囊中了，现在你已经把使 
用 XML 来以多种形式表现你的页面的能力加入 
了你的工具箱。 




二 r 

■，減 一 \ 

•板 



9 REST 乌 Ajax 


命 


，更上一层楼 



是时候巩固你的混搭技术了。到 U 前为止你已经行过如何添加 Google 地图判 
你的 Web 应用中 来清晰 地展现为空间数椐。但蛙如*你想要扩展已经存在的功能 
呢？淸继续读下去，我们会为你展示如 W 添加更先进的 Ajax 精华到你的混搭式应 
用 （ mash - ups ) 中。 ifiiH ., 你会通过这种方式学习到史多 REST 的知识。 
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滑坡，还有什么？ 


事件太多了 .『 

随畚用户界面的改进，访 NHeadFirsiaimbcrsM 站的人数急剧举 Jh 但麻烦的是，太 
多的事件 ( incident) 记录下来以至 T 人们很难把它们都看完。 



网站的索引页面使用两种方式来显示信息。 


o 页面顶部是用纬度和经度记录的详细的事件列表。麻烦的迠，很多人需 

要翻过 这部分内容才能到达 5 r 面底部的地围区域。 

O s 你点击某个亊件时， m 贞会把哉减后的洋细 信息显 示在一幅地 ra 上。 

这儿的 mis 在干不是所存的数 据都显 示在这幅地图 t 。 

I 

这两茗都+是那么完全让人满意。用户很难从列表中定位供件，而这正是我们添 
加地图的原因。 m 迠地图并没右显示所有的可用数据。那么我们该怎么办呢？ 
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REST 与 Ajax 


M 礅玎认显示烫多的洋细信息 

完美的解决方案足 u ： 地 m 做更多的私汁。如果它能够显示事件的吏 
多冇用信息，我们就可能可以 i ; •掉事件列表而 U ： 整个主贞变成一个 
多功能的大地囹。这就意味肴我们不需要通过多个页面来显 示史多 
的数据。 





还有一个 H 题： 地图局部模板的代码已经下载好了。它简单易用， 

但是我们兴要修改这段代码吗？幸运的是，地图局部模板使用了一 
个开发技巧使得我们不需要触及下栽的代码。但这个技巧是什么呢？ 

对于我们不用修改下载的代码就能更改地图，你是怎么想的？ 
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google maps ； lajax 马 迗动的 


我们能够扩 展基子的地穆 

编 v 地 ra 局部校板的人觉得人们很快就会希望扩展地图的 工 怍方式。 
由 F 地图局部模板调用 Google Maps . 而 Google Maps 又足使用 Ajax 建 
>>:的，这就使 得通过 Ajax 来扩展地 ffl 成为可能。 


地阁的 I :作方式是通过向服妗器端 
发送淸求来获取一个 ti 含系统中所有饫山半什. 
U 1 录的 I 羊细信息的 XML 文件。默认情况下，地 
阁会 W . 冶 XML 屮乜含 的知题 和描述。 

仉坫地 ra W 部校板也允 it •你 f 专人诂示祐个 ‘ K ： 件 
的的动作名称。这个动作可以 坫任 H 你想 
耍的动作,所以如果你想意,你按至可以生成 
类似于事件 show 页而最初版本那样的洱面„ 








我们要 il ： _m a p . html. erb 局部生成 Ajaxifi 求，这就意味荇 
它盅要 WHProioiypc 库。我们可以通过与前面-忭的方法来实现 
这点， Hi 就是在布 W 义件中添加如下的対 JavaScript 屮的引用： 


<%= stylesheet_link_tag 'scaffold* %> 

<%= auto 一 discovery 一 link 一 tag( : rss, { : action=>'news'}) %> 

<%= javascript_include_tag 'prototype' %> 

</head> 
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REST 与 Ajax 


但我们怎#才能改 变索引 页 i 哝？ 

吏改主 页所要做的访一件氷件躭圮刪除 亊件 列丧来 u •.地 a 变得更 
大： 



我 m 还需奘: 1 ;•知地 mw 部模板用来 a , r < 杯怍位息的动作名称。所以 
变化还是有一些的，但实际上它会让 index . html . erb 模板比以 
前商电很多。 



本次:上，下曲'娃我们将会留 F 的 内容: 


当地局部校板通过这种力-式被调用时，它就改变了它的行为. 
以前，当一 件荠 伴被点击时，局部模板运行一段默认的 JavaScript 


来仵弹出的信总窗 U 中泌 示鉍 题和描述。这个改动表明，地图 上的 


某件 唭件 被点 A 时，局部模板将调用 show 动作汴在窗 n 中敁示这 


个动作的响应。 


或者说，一旦我们让 show 动作生成正确的输出，它至少就会这 
样。 
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显示更多的信息 


“ show ” 动作熏要生成怎样的沟容? 


我 fl 】 ti 经有一个 Show 动作了，它生成包含丰怍详细信 
息和事件的位 S 地图的网页。 

但现在我们不需要这么多内容了。 Show 动作 H 需要生 
成事汁•的文本详细倍息就行，由于信息将在地图上的 
某个点被点 Ji 之后显示，所以我 fN 就不用 W 显示纬度 
和经度。 

还冇一个不间的地方：我们只需要•个页面片段。我 
们不需要由页面模板生成的标准 HTML 样板。这就意 
味眘我们的动作需要从局部模板中生成。我们把这个 
W 部模板取名 S _ show.html . erb 。 



，令、残 们芍以 眺过鋅嗄扣廷度— 
用户含釦迮6赛的。 



我们不 tf 池® 




上阵 


请完成控制器的 “Show” 
html.erb 局部 模板： 


def show 

@incident = Incident.find(params[ : id】> 
respond 一 to do |format| 


方法里的空行来实现调用 _ show . 



format.html { 


} 

format. xml { 

render : text=>@incident.to_xml( 

: only=>[ : latitude,:longitude, : title, : description], 
: root=>"data") 

> 

end 

end 
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请完成新的 _ show . html . erb 局部模板的代码。记住 你不需要 
显示所有的信息。 o _ 


rl incidents 



_show.html.erb 


incident.longitude 




incident.title j 



incident.mountain 


■ 


incident.latitude 


J^Descri^ti^n^ ^Title^^ 
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解决你的问题 


、 V 


笔上阵 
解答 


请完成控制器中的 “Show” 方法里的空行来实现调用 _ show . 
html.erb 局部 模板： 


app 



def show 

@incident = Incident.find(params[ : id】> 
respond_to do |format| 
format.html { 

rtv^Xtr rpartuil= > sWow' , : U>frflLs=>{ ： l*^cidc»M:=>@u^6tdd»^} 

> 

format .xml { 

render : text=>@incident.to 一 xml( 

: only=>[ : latitude, : longitude,:title,:description], 
: root=>"data") 

> 

end 

end 


ontrollers 

incidents_controller.rb 
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^ 代碚贴冰葙糍铁解著 


请完成新的 _ show . html . erb 局部模板的代码。记住 你不需要 
显示所有的信息。 ^__ k 




incident.mountain 



incidents 

31 _show.html.erb 


^3 ..…… </b> 

incident.when ^ 


JlW itl |Ti ： n l |l..</b> 

1 incident.title | 


I Description: I 

<b> I. . </b> 

^^c^en^^escriptior^J 笔〉 


incident.longitude 
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为大地图唱彩 


— 霜- 

现在如果我们来到索引页面，可以确定的是 —— 事件列表已经消失而 
地图变得更大了。更重要的则是当用户点击地图 h 某个本件时发生的 
变化： 



当地图检测到某个亨件上的鼠标点击时，它会生成一个 Ajax 请求给 show 
动作，而这将生成事件的 HTML 格式的详细信息。地图会接收到 HTML 
并用它来替换事件的信息窗口中的内容。最后信息窗口被显示给用户。 
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新的 她懷功 能成功了 


新的地图功能嬴得 f 登山者们的热烈欢迎。他们不再需要翻过很 
长的价 t •列表>1•能来到地阌区域。现在他们能够立接从地 m 上获得 
他们耨要的所有信息。只刺 F —件单件 T …… 



登山者们想要使用地图来报告新事件。 

他们能够使 ITIGPS 设 务来找 到他们的纬度和经度 
然斫把这呰位息输人， m 坫如! I !：他们能够 k 接点 
山地 m F : 的某个点并在那儿填3其他详细信息， 
这个网站就更加》用广，那样，«山荇们的数据 
珙人就会快很多。 


在地图上输入数据与我们现在实现的 
方式比起来如何呢？ 







异步请求 



如果有人想要创建一个新的事件报告，当前他们必须经过如下几个步骤 


点击首页上的新事件链接 （Newlncident link ) 。 

你无法在首页上直接输人数据，相反，你需要通过一个指向 “new” 
页面的链接。 




在 New 页面上手工输入纬度和经度 a 

这*页并没存地图，所以你需要手丄输人纬度和经度汴保存记录。 
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你创建的事件被显示出来。 

一旦你点击保存按钮，你就被重定向来到了我们刚刚创逮好的缩小 
版的 “show” 页面。如果你需要创建第二个事件或者回到主地图， 
你就需要多次点击后退按钮来到首页 一一 在那儿你可以重新开始所 
有的步骤。 



邡 么哪些 t 要傲修改哝？ 

与遍历上述所有步骤不同的是，用户希望界面 "1* 以吏加 简单。他们希 
望直接在地图上进行点击并使用弹出窗口中的表单来填写详细信息。 



为了灾现这一点，我 ffl 需要使用 Ajax 来生成 “ new” 表申.。我们还 
盂要在冇人点,1?地 [1 h 某个新位置时让地 ffl 来调用这个表单。但怎 
样实现呢？ 
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render 可以有多个动作 


M © 爲部糢板玎认让我们指定 
-令 “ new ” 动作 


目前为止，我们已经看过如何在地图上 a 示亊件的详细信 
息。但是我们应该如何创违新事件呢？ 

_map • html. erb 局部揆板允I午我们指定一个动作来处理 
新事件，这与它允许我们指定显示亊件洋细信息的动作一 
样。这躭意味着我们可以如下在 index •htrol.erb 文件中 
添加一个 “new” 动怍，就像下向这样： 


app 


1 ~> I views 

• ―> 厂 I incidents 

|_v j^J index.html.erb 


<hl>Listing incidents</hl> 

<%= render (: partial*>'map', :locals->{:data=> M /incidents.xml", 

: ful^page^true, : show_action->* show 1 , : new_action=>' new' }) 


# 饩輋'主成他哆 i 


•抑糾 ❼以十糾奶; 


如果有人在地图上点击一个新地点， _map.html.erb 将创逮.个 
新的 fc 记并弹出一个信息窗口， 该窗口 位含 "new" 动作返回的内 
容。 

我们已经有_-个为应用定义的 “new” 动作，但是它生成的是整个 
网页。我 (H 不需要 S 示整个网页，而是需要 “new " 动作来创建 
个将被显示在弹出信息窗口里的页面片段。我们也需要确保当用户 
提交 “new” 表申.时，表单依然停留在地 ft) 上。 

所以我们将创建一个名为 _new.html.erb 的局部模板来生成一个 
Ajax 表单。 


370 第 9 章 





REST 与 Ajax 



你需要为表单创建 _ t _ new . html . e rb 局部模板。请完成该表单的代码。记住 —— 用户不 
需要输入纬度和经度的值，但是表单依然需要记录这些值。 


<hl>New incident</hl> 

<% remote_form_for(incident) do |f| %> 



<P> 

<%= f. submit "Create 1 ' %> 
</p> 

<% end %> 


当地图调用 “ new ” 动作时，它把新事件的位置信息作为请求参数发出。请完成 incidents 
controller , rb 的 new 方法中的代码，使得它能够正确地调用局部 桟板： 


format. html { 

@ incident.latitude=params['latitude'] 

@incident.longitude=parcuns['longitude'] 


} 
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调整斩的调用函数 



你需要为表单创建一 t„new.lnml. er b 局部模板。请完成该表单的代码。 记住一 用户不 
需要输入纬度和经度的值，但是表单依然需要记录这些值。 


赚 

<hl>New incident</hl> 

<% remote 一 form 一 for(incident) do [f| %> 

<?> 



<?= : kvu > ukvtalkv ^> <多 =: ^ 0 > 

^ .. 斟*聋中的! 


= f.|auddc^_fUlc4 : y ^ 踔電 it 它 fl 速饩 均核 

< ? = f ^itu.d« ^ 说访存 


~a 


ncidents 

new.html.erb 


<f= f.lattl ：wlncw ^> <^ =■ f.c< ： wh«^v ^2 
</^> 


<J = f.Label : tltU J> <J= f.Uxt.fteUI : titk J> ^T" 

.〈.今〉 . 


你的代 《 g « 金 
'车哆 不阌. 


< 多 =: f.label : description f><tor/> 
- : desmptu)^, : rows= 


</p> 

<p> 

<%= f.submit "Create" %> 

</p> 

<% end %> 

当地图调用 “ new ” 动作时，它把新亊件的位置信息作为请求参数发出。请完成 incidems 」 x ) mroller . r b 的 
new 方法中的代码，使得它能够正确地调用局部 模板： 

format.html { p >. ， 

@ incident.latitude=params['latitude ’ 】 1 - j 3 叩 

@incident • longitude=pararos 【 1 longitude • 】 Icontrollers 

rtv^dtr rp«rtial= > r v^tvj' , ： lccals=>{ ： i^lde^=>®l^ldc\^Jt} K ... 

.. jv |9r| incidents_controller.rb 
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现在当用户来到首页并点击地闬上的新地点时，他们能够使用-个弹 
出表黾来输入详细信息。当 “ Create ” 按钮被点击时，表羊依然怜留 
在屏孫上，但是一条新记录被兄示在数据库中。 



m 


.683975 

-78. 4365055… 

L. 123925 

72.72135833 … 


那么是不是一切正常呢？ 
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混乱是大忌 



虽然丧单 iK 常： u 乍而 a 事 汴也被 保存到广数据痄中, m 是搴实上肴起 
来什么事都没有发生 # 当用户点击 “ Create ” 按钮时，并没有任何反 
馈来表明 i 己录被保心 •• f 。这就盘味行用户会小断点 〖 tfCreat 按钮，⑻ 
数据 Hi 将收到很多条农的 id 洎。 



需要做•些修改。 fr : 我们刚刚使川支架头现应用的那个时候,，用户 
使用 “ new ” 饭曲报告•个事件时，浏览器将立即跳转到 “ show ” 页 
而。这会确认数据被保到 r 数据床。 


我们可以在 Ajax 应用中有类似的东西吗？ 



Ne \ 
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我们如何证硪一件擧件 B 经狨保存？ 

系统确实崙要使用 “ show ” 动作在弹出窗口中显示创建的 id 录。所以 
如果有人输人某个奉件 的详细 信息，弹出窗 U 将变为显示这个事件的 
只读版本。 



在这沖方式下，弹出的信息窗口将像一个小型浏览器一样工作 , am 
到新的唞 ftf / i 息。3然我们的代码小能 ill 浏览器前进 到新馆 息豇面。 
我们需要 It 登山者停留在当前页……只是能够到显示的新信总。 


—- 

Ajax 表单蒲要被事件的 “ show ” 动作返回的内容所替换。你认为你如何才能够 
实现它呢？ 
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就地更新 


表单 f 要 E 新弹出窗 o 的 < div > 的沟 
容 

虽然 Gtxigle Maps 看起来儿乎就是个桌面应用， fK 它基本上还是 ijH 结为 
HTML 和 JavaScript 。 它就是一个 MM 。 这意味咎弹出的仏息窗11 
以及其他所有内容-都 H & HTML 片段。 

弹出窗 11 的内容通过如下代码彼定义在 <div> 元素 m: 


id=' map 一 inf o * 

这很重要，因为我们使用 Ajax 表承 来创达 •个 新事件报告，而 Ajax 表 
单可以被用来使用它们的 id 动态史新甸部豇面。 



所以，如果我们能够让表单使用杯件的 “ show ” 动作返回的内容更 
新 map _ info 这个 < div >, 网页就能够给予用户他们所需要的反馈。 
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这是 “ new ” 表单的部分代码。记住，表单需要使用表单的 
响应内容来更新页面，请完成代码： 


)do |f| %> 


<%= f.label : mountain %> <%= f.toxt_field : mountain %> 
</p> 


“ new ” 表单把自身提交给 “ create ” 动作 • 这是控制器上 
的 create 方法。请标出你认为需要的修改。 


def create 



@ incident = Incident.new(params[ : incident]) 


respond 一 to do |format| 
if @incident.save 

flash[ : notice] = 1 Incident was successfully created. 1 
format.html { redirect 一 to(@incident} } 

format.xml { render : xml => @incident, : status => : created, 
: location => @incident > 

else 

format.html { render : action => "new" } 
format.xml { render : xml => @incident.errors, 

: status => : unprocessable 一 entity ) 

end 

end 

end 
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没有任何修改: 


k 磨笔上阵 
蘇答 



这是 “ new ” 表单的部分代码。记住，表单需要使用表单的 
响应内容来更新页面，请完成 代码： 


incidents 


w.html.erb 


<% remote form for(incident, 




do |f| %> 


<P> 

<%= f. label .-mountain %> <%= f. text__field :mountain %> 
</p> f2| 



app 

controilers 

mcidents_controller. rb 


“ new ” 表单把自身提交给 " create w 动作 „ 这是控制器上 
的 create 方法。请标出你认为需要的修改。 

def create 

@ incident = Incident.new(params [: incident]) 


respond 一 to do |format| 
if @incident.save 

flash[ : notice] = 'Incident was successfully created.' 
format.html { redirect_to(@incident)) 

format.xml { render : xml => @incident, : status => : created, 

: location => @incident } 

else 

format.html { render : action => ’ 'new” ) 
format.xml { render : xml => @incident•errors, 

: status => : unprocessable_entity } 

creati : 劫 n 中；沾表掌 li 祛 i 时.记录接 
•M‘) 盎荈 速中 .* 后 祀璜求 fife f_) "show - 茶 "show- a 

n 现 <54 滅的 4S- 子靶搴 4 的镱 .6 的 HTMU4a„ ias«S*i4« 


end 

end 

end 
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那么现 ft 当用户创违一个事件时会发生什么呢？ 

在地图 t 点 it ? 某个新地点就会像以前一样显示 Ajax 表争: 



但是当用户点击 “ Create ” 按钮时，系统不仅会将记录保存到数据库 
屮，而它会返问•个 fei 含事伴的详细信总的页面片段， Ajax 表电会 
使屮 这个! SlAi 片段柬史新弹出窗丨丨中 map_info < div > 的内容。 



<如>饵妁 "5 更扣。 
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没有蠢问题 


f 没有 & 

没存蠢网题 

产没有蠢输 


1®)' 如果用户的浏览器不支持 JavaScript 会怎样‘？ 

地图应用将无法运行。像 Google Maps 这样的 
Ajax 应用需要 JavaScript 。 

问： 如果 Google Maps 是 Ajax 应用，那为什么上一 
章我们不用引入 Prototype 库？ 

• Google Maps 调用所有来自 Goolg 服务器的 Ajax 
库，所以它不需要 Prototype 库。 

1^)* 那为什么这次我们需要引入 Prototype 库？ 

如果你把动作名称传递给地图局部橫板，那么 
这个局部橒板就需要发送 Ajax 请求给服务器 e 这样它 

就需要 Prototype 專 . 这部分处理与 Google Maps 是无 

关的 s 

有没有什么地方在用户禁用 JavaScript 的情况 
下让应用脱离地图运行？ 

如果你修改控制器的话就能做到。控制器决定 
了显示哪一个视图，所以它能够在没有 JavaScript 的情 
况下运行不同的页面模权和局部模板。 


•蠢 河题 

我还是不明白 respond_to 是如何工作的。如果 
说 format.html 是一种方法调用的话，在它之后的 { • } 中 
的代码又是怎么回事呢？ 

* 在 Ruby 中，方法可以把 {•••} (或者 do ... 
end ) 中的代码片段作为参数来看待。所以在{和}之 
问的代码会作为参数传递给 format . html ， 而 format.html 
則决定了是否运行这段代码。 

地图局部模板是如何工作的？ 

它并不是那么复杂，但是它大多榖都是 
JavaScript . 所以在这儿我们并没有讨论太多的细节， 
不过你可以查阅 《深 入浅出 JavaScript 》 来了解更多这 
种类型的内容。 

但是我真的想知道它是如何工作的！ 

:你最好 W _ map . html , erb 文件看起。如果你 
想要了解更多我们所提及的 Ja vaScripi 《深入浅出 
JavaScript } 是本很不错的参考书： -> 







编辑功能如何实现？ 

规在編辑是如何实现的…… 

支架在两个地方为你提供编辑选项。 

在原先支架版本的 “ index ” 页面中，你可以点击任一记录之后 
的 “ Edit ” 链接来跳转到编辑表单。但是现 ft 我们无法这么做了，因为 
索引页面的事件列表已经被地图所取代。我们知逬地图局部模板并没有 
内置的 “ Edit ” 功能 0 



耶么在原宋的支架版本中我们还能在哪儿编辑事件呢？好吧，另一 
个地方就在事件的 “ show ” 页面。在应用的 i 架版本里，在 “ show ” 
豇面中有个 “ Edit ” 链接。 



我们可以使用类似的方式来实现编辑功能吗？我们是否可以考虑在弹 
出窗门显示的事件内容里添加一个 “ Edit ” 链接呢？ 

这样可行吗？ 


382 第 9韋 











还能有多难? 
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REST 与 Ajax 


我们玎认在缂出 f 口中加入一个 “ Edit ” 链摟 


我们需要 fti 羊细信息中添 加一个 “ Edit ” 链接，而这个 详细佶 息将在有人选屮地阁上的 
某个事件时显现。当有人点击 “ Edit ” 链接时，我们把 mapjnfo <div> 的内容替换成 
陡示一个编辑表单，然后用户可以使用这个表单来修改 id 录。 


我们已经有 f 实现好的 ' how ” 功能。 “ Edit ” 表中.与 “n 
多.而且我们也有控制器中用来修改记录的服务器端代码。 


丧申.羞不 










两个局部模板. 


我们从修改 “ edit ” 动作孖始 



为 f 十么要 冇两个呙部栈板嘹？ 

这两个局部模板基本上是一样的代码，但保持两个局部模板是 
比较好的做法。目前它们看起来一样，可是将来就不—定 r 。 

比如，我们可能会修改 “ new ” 和 “ edit ” 页面所提供的功能。 
我们吋 能会在用户插入一个亊件数据后就不再允许修改。我们 
也可能把两个页面的外观做得完全不一样。 


cdit . litiad . crbfo 从 w . IntwX . eri !) 中的 
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Rails 能够知道表单的模型对象之前是否被保存到数据库。 

所以 form _ f 0 i : 辅助函数生成的代码会根据它所处理的是没有保存过 
的对象（这种情况下，表单将调用 “ create ” 动作） 还是已经被保存 
过的对象（这种情况下，表单将调用 “ update ” 动作）来决定如何修 
改。 

除了创建局部模板，我们还需要修改控制器中的 “ edit ” 动作方法以 
在它被调用时返回 “ edit ” 局部模板： 



def edit 

@ incident = Incident.find(params[ : id]) 

render :partial=>'edit*, : locals=>{ : incident=>@incident) 


Rails 如何知道一个对象是 
否已经被保存过？ 

I 它调用一个名为 “ new _ 
record ?” 的方法.这个方法在对象 
从没被保存过的情况下返回 ture ， 

1^) • 为什么 “new_record?” 方 
法在末尾有个问号？ _ 

这是 Ruby 的规范.大多数 
返回 ture 或者 false 的方法都会在它 
们的名字中包含一个问号. 


v 々存蠢问觀 

为什么我们必须要有纬度和 


经度的隐藏域？ 


我们不希望用户去编辑它 


是一我知道这一点。但是 
为什么要在表单中提到它们呢？ 

表单对象会转化成表单中的 
域。如果它们没有在表单域中出现， 
它们就不会出现在记录中。 


卜 “ 但我们不是也没有 
id. created_at 和 updated_at 域吗？ 

确实——但 Rails 知道需要这 
几个域，所以 form _ for 辅助 A 数会 
为你创建它们。 


你现在的位置 ► 


385 




ajax 风格的编辑 


我们还 f 要在 show 页 i 上缯加一个 
新 M 狻 

我们现在应该可以生成编辑表单了，但是用户怎样才能到达编辑表争 
呢？我(1飞需要让用户在查看事件的详细信息时也能看到一个 “ Edit ” 
链接。这个链接需要被添加到_3匕0>/.111：!111.6比局部模板。 





Listing incidents 


Listing incidents 


我们将使用 link_to 辅助函数朿 •生成 链接。 
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那么我们该如何使用 link 一 to 箱助&数? 

link to 辅助函数有两个 参数： 链接的显示文本和链接的目的地。 


<p><%= link 一 to "Edit ”， "/incidents/#{incident.id}/edit" %></p> 


好吒.轼达 祥吒。 我 B 轻受够了达螫我们一 i 在值用 
的古饯的路技。为什么我们必铽抬入达整长字符宰？ 
我们不 JIB 轻在路谂中定义7路技喝？难道我不守认盗接 
说“达是 incident 对象的 edit 路枝”或类似的东®吗？ 


这是很不错的建议。 

迄今为 止我们已经使用字符串来创建了大 t 的路径和 URL。 但如 
果我们将来希望改变链接的格式会怎样呢？我们可以很快就改好路 
由中的内容，但是我们的代码中有大 M 冗余的包含路径的字符串。 

在两个地方拥有相同的信息不是一个好 匀惯， 因为它破坏了一条重 
要的 Rails 原則: 

一 以镝不基'兄 

>要重箕你乍己 

不过如果路由已经记下路径和 URL 的结构呢？也许我们最好先仔 
细研究一下路由。 
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REST 风格的路由 

JB 故;柃的摊由 - 

0前我们使用 map • connect 命令在 config / routes.rb 中创建了多条路由: 


map.connect ' incidents / map / : id ' , : action =>' show _ with _ map ', : controllers * incidents • 

不过当你编辑由 Rails 支架生成的 routes . rb 文件时，你可能已经注意到它里面 
的路由使用了一个不同的 命令： 

map.resources : incidents 

这个命令生成一组被称为 REST 风格 （ RESTful ) 路由的标准路由。这些路由 
使得用户能够访问应用的 CRUD 操作。你可以使用 rake L 具在控制台上检验这 
些路由: 




incidents 

formatted 一 incidents 


new 一 incident 
formatted_new_incident 

edit__incident 

formatted_edit_incident 

incident 

formatted incident 


GET 

/incidents/news, xml 
/incidents 

GET 

/incidents.:format 

POST 

/incidents 

POST 

/ incidents. : format 

GET 

/incidents/new 

GET 

/incidents/new. : format 

GET 

/incidents/:id/edit 

GET 

/incidents/:id/edit•:format 

GET 

/incidents/ : id 

GET 

/ incidents/ : id.:format 

PUT 

/incidents/:id 

PUT 

/incidents/:id. : format 

DELETE 

/incidents/ : id 

/:controller/!action/'id 


controller^^incidents", 

controller- 〉 "incidents", 

controller ， >”incidents”, 

controller->"incidents ,, r 

controller->"incidents", 

controller=>"incidents M , 

controller-> w incidents ,, # 

controller=> M incidents"# 

controller=>"incident3% 

controller- 〉 ”incidents", 

controller»>”incidents", 

controller=> H incidents", 

controller=>"incidents" / 

controller-> M incidents , *, 


：action*>"news"} 

: actiondndex”} 

: action->"index") 

: action^> M create w } 

: action=>"create w } 

: action»>"new w ) 

: action=> M new") 

: action->”edit"} 

: action-> H edit M } 

: action=>"show*M 

: action=>"show’’} 
:action=>"update*M 
: action-> w update"} 

: action=> M destroy") 


每一行都是单个路由，而有些路由取了名字。而 edit _ incident 则是 
“/ incidents /: id / edit ” 路由的名字。 

但这又是如何帮助我们清理代码中的路径的呢？ 


388 第9章 





REST 与 Ajax 


Rails 为每个命名路由 (named route ) 槿供？辅助&数 

这些 REST 风格路由的名字至关甫要，因为它们将帮助你在代码 里指向 特定的 
路由。它们允许你把如下的 路泾： 

•’/ incidents / incident . id } / edit n 

改成： 

edit 一 incident 一 path ( Qincident ) 

对于毎个命名路由， Rails 都给予你相应的辅助函数来生成在本地服务器上的 
路径和完整 URL 。 

在本地服务器上的路径 

edit _ incident _ path (@ incident ) returns / incidents /3 /edit if @incident has id = 3 

完整 URL 

edit 一 incident_url (@ incident ) returns http : //localhost : 3000/ incidents /3 /edit 

它们被称为 RESTW 格的路由辅助函数，因为它们能够把 资源成 者模型对象作为参 
数。记住—— REST 设计的一条原则就是把 Web 应用当作资源容器。 

在调用 incidents 和 new _ incident 的辅助函数时无需传人资源参数 比如 

incidents _ url 和 incidents 一 path * 

路由辅助函数不仅可以从你的代码中刪除冗余路径，而 a 它们 更容易 阅读并会减少 
输人错误路径的可能性。 

那么这会给我们的代码带来哪些变化呢？它会把这段 代码： 

< p ><%= link 一 to M Edit", "/incidents/#{incident.id}/edit" %></ p > 

改成： 

< p ><%= link_to " Edit ", edit_incident_path(incident) %></ p > 
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考, ertU 铯鴉戍 旮出现 
在*«携板 i "5 


Dcwrtpuon 

0>oflptd by cinp «c«>O9f0» 
«*!'>« lou i410 Mi» 


Listing incidents 


<0^, 到虼器铯利 
© 一个否 $ 条 *5 _ 


这个链接 a : 接把浏览器带到 r 


一旦 “ Edit ” 链接被添加到 “ show ” 局部模板中，我们再次点 * 地图上现 
冇的 *1^1 •会发生什么呢？ 


http : // loca - lhost : 3000 / incidents /5 /edit 

问题是，现在这个链接把_ 60 ^ 1 : . html . erb 局 部模板 的内容送回给了浏 
览器，这看起来就像在另外一个豇_ H 

我 ffn 范要泠 留在浏 览器的1前 lK ，那么我们怎柞修 正它 《足？ 
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让 Ajax 链狻來牷救我们 

我们前面创建的 “ new ” 表单能够#换弹出信息窗口内容的原因是 
它给服务器发送了 Ajax 请求。它使用服务器的响应来替换 ma P _ 
info < div > 的内容。 



但是我 ffl 刚刚添加的链接件+像这样运作。它仅仅告诉浏览器链接到另 
—个 K 面。如果我们创达一个 Ajax 链接而不是浏览器链接的话，我们就 
能解决这个问题。 

Ajax 链接的工作方式与 Ajax 表单很像。当你点击一个 Ajax 链接时，它并 
不会让浏览器跳转到另一个页而，相反，它会生成一个 Ajax 请求给服务 
器并使用返回的响应朿更新页面的局部内容。如果这个过程看起来很熟 
悉的话，那是因为 Ajax 链接与我们前面用于刷新邴子航空座位列表的 
Ajax 按钮几乎-样。 

我们需要做如下修改来把链接转为 Ajax 链接： 

<p><%* link_to "Edit", edit_incident_path(incident) %></p> 


变成 这样: 




< p ><%» 1 ink_to 一 remote .’ Edit ", : update => "map 一： 

: url=>edit_incident_path(incident) %></p> 


in^T" 






现住这个链接应该能够从服务器那儿生成一个编辑表单并在弹出窗 
口中显示它 r 。 让我们#看现在它是怎样工作的。 
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当我们点占一个事件时，信息窗〖1与以前究全一样。 


链接否 fe 来也一样，但足请记住实 际上它 不再是一个简申的链接。相反, 
这儿有大饿 JavaScript 的噔法在起作用，等右在链接被点击时来产生一个 
Ajax 沾求。那么 ’*1 我们点 ifi 它时会发生什么呢？ 


我们期嗜的编辑表单并没冇出现，相反，我们得到 r 这个办怍的“末知动 
作 （Unknown action > ”错误。究竟发生了什么? 
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我们用铐？踣由 .r 

当 Rails 接收到链接发送过来的 Ajax 请求时， Ajax 链接发送了正确的 
请 求至： 

http :// localhost : 3000/ incidents /5 /edit 

但 Rails 并没有把这个请求匹配到 edit _ i nC ident 路由，相反，它 
匹配了一条默认 路由： 


incidents 

formatted incidents 


new_incident 

formatted_new_incident 

edit_incid«nt 

forraatted_edit_incident 

incident 

formatted incident 


/incidents/news.xml 
GET /incidents 

GET /incidents. : format 

POST /incidents 

POST / incidents. : format 

GET /incidents/new 

GET /incidents/new. : format 

GET /incidents/:id/edit 

GET /incidents/ : id/edit. : format 

GET /incidents/:id 

GET /incidents/ : id.:format 

PUT /incidents/ : id 

PUT /incidents/ : id.:format 

DELETE /incidents/ : id 

/ ： controller/ : action/ : id 


{ : controller=> w incidents", 
.{ : controllers"incidents", 

{ : controller^"incidents", 

{ : controller»> M incidents", 

{ : controller-> M incidents" / 

{ : controller-> ,, incidents M , 
( : controller*> H incidents**, 
{ •• controllers" incidents", 
< : controller- 〉 "incidents", 
controller- 〉 ”incidents ”， 
{ : controller=> M incidents M , 
{ : controller=>"incidents ", 
{ : controller-> ,, incidents", 
( : controller->"incidents", 


: action*> M news M } 

: action=> H index"} 

: action-> w index w } 

: action->"create w } 

: action->"create M ) 
:action-> M new M } 

: action->"new") 

: action*>"edit") 

: action»>"edit"} 

: action->"show w ) 

: action»> w show") 

: actiorj">>"upciate"} 

: action^> M update"} 

: action-> M destroy w ) 


衫 e e f 路 f . 
系不袅 t i ： 面的 eckt 路® 


Rails 尝试把它与綠近底部的默认路由进行匹配， ： action 参数被设 
为 “5” ， 而: id 参数被设为 “ edit ” • 伹并没有名为 “5” 的动作，所 
以它失畋了。 

怎么会这样呢？我们的 URL ( hup :// localhost :3000/ inciden 〖 s /5/ edit ) 与 
edit _ incident 路由 (/ incidents /: id / edit ) 具有相同的路径格式。它 
们怎么会没有匹配成功呢？要知道，在我们把链接转成 Ajax 之前还是 
它 T ： 作得好好的。 


—切妨- 

请再次查看路由列表。原来的链接和 Ajax 链接指向的是相同的 URL 。 你认为是什么原因 
导致 Ajax 链接匹配了错误的路由呢？ 
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匹配更多的路由 

H 1 TP 方法是迭择路由的一个哆素 

在路由表中有一列我们没有注 意到： 

看么 …… 






/incidents/news.xml 

{ : controller->"incidents ”， 


GET 

/incidents 

{ : controller=> H incidents", 

£ormatted_incidents 

GET 

/incidents. : format 

{ : controller-> H incidents". 

POST 

/incidents 

{ : controller=>**incidents"# 


POST 

/incidents. : format 

{ : controller->"incidents", 


GET 

/incidents/new 

{ : control ler ss > , 'incidents", 


GET 

/incidents/new •: format 

[•• controller** 〉 "incidents "， 

edit incident 

GET 

/incidents/ : id/edit 

{ : controller=>"incidents ”， 


GET 

/incidents/ : id/edit. -.format 

{ ： controller»>"incidents M , 


GET 

/incidents/rid 

{ : controller=> w incidents", 


GET 

/incidents/ : id. : format 

( : controller=*> M incidents' 1 . 


PUT 

/incidents/:id 

{ : controller->"incidents'% 


PUT 

/incidents/:id. : format 

{ : controller- ， incidents". 


DELETE 

/incidents/ : id 
/ ： controller/:action/:id 

{ : controller^"incidents", 


那么这些 GET，POST. PUT 和 DELETE 是什么意思呢？ 

它们是 HTTP 方法-也叫做 HTTP 动词 (HTTP verbs ). 
毎 个谪求 都会使用一个特定的 HTTP 方法，而 Rails 使用泫 
方法和路径来决定选择哪条路由。 

但它们究竞代表什么呢” 


: action»>"news’” 

: action »> ,1 index*' ^ 

: action* 〉 ”index”} 

: action«>"create M ) 

: action->"create"} 

: action*»>"new"} 

: action»> H new") 

: action»> w edit"} 

: action •: >"edit"} 

: action*> w show M ) 

: action->"show , *} 

: action=>"update" | 
:action->"update*M 
: action=>"destroy H } 
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什么是 HTTP 方法? 


不考虑名字的话， HTTP // 法与你听能找到的任何 Ruby //法，比如 
控制器的方法,没冇任 H 相似之处。相反， HTTP 方法使用在客户 



为什么两种版本的链接幺做不同的 餐什 呢？ 奸吧 ft 通的 HTML 
超链接 （ hyperlink ) 发送 GET 请求给服务器。恨垃，默认情况 
F , Ajax 链接发送 POST 请求。 


所以为 HI : 这个链接能够£常 T 作，我们还需 要让仑 使用如 F 的 
HTTP 方让： 


<p>< = link_to 一 remote "Edit", : update *> n map_info", 

: url»>edit_^incident url (incident), : method=>'get' %></p> 
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(再来一次）试驾 





当我们点击事 ft 链接，我们#到 r 带有 " Edit ” 链接的倍息。当我们点击 
这个链接，它创让了 •个 AjaxW 求来获取编辑表单，而该表单被 fH 来锌换 
弹出窗 II 的内容。 
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t ^). a 使用字符串作为路径真的那 
么糟糕吗？ 

字符串可以工作，但是它们 
睢以被读者 理解. 而且比起辅助函数 
来说史容易产生错误。 

为什么会更容易产生错误呢？ 

答： 如果你拼写错了一条路由辅 
助系数的名字. Rails 会报告一条播 
误给你„但如果你在字符串中拼写错 
了一条路技.系统无法报告该路径错 
谈，也无法报告由于该错误路径而引 
起的其他铋误 3 

I '®) • 为什么 link _ to_remote 创建 
了 POST 请求而 link_to 却创建了 GET 
请求？ 

link _ to 刨建了一个简单的 
HTML 超链接。浏見器总是为阕单 
超链接使用 GET 方法 • 但是 link _ to _ 
remote 创建了 一个 Ajax 请求，而 Ajax 
请求默认总是使用 POST 方法来提交。 


—— t 没夯 蠢码题- 

蠢碰 

1^)* 为什么 HTTP 需要考虑使用 
GET 还是 POST ? 这么做有什么好处？ 

GET 请求被设计成可重1的。 
所以无论你发送多少次相同的 GET 请 
求都没有关系. GET 请求通常用未读 
取信息。但是 POST 被用于每次都可 
能修改报务器上軚据的请求里.所以 
它们通常用来更新啟 据库。 

问：那么 PUT 和 DELETE 又是什 
么呢？ 

PUT 被用于在 ft 据库中刨建 
新记录的请求而 DELETE 則用来刪 
除钕据 库中的记录， 

I '®)- 这对所有 Web 应用都成立吗？ 

它适用于 Rails 应用 使 
用正确的 HTTP 方法是 REST 风輅 
( RESTful ) 的设计非常重要的一环， 

1^) .* 所以这就是为什么 form_for 
能够使用相同的代码来生成可以更 
新或者插入的表单？ 


答： 是的。如果对象已经被保 
存， form _ for 会生成使用 PUT 动作的 
表单如果对象是新的，那么它会生 
成使用 POST 的表单。 


问: 


有人告诉我浏览器不能使用 
PUT 和 DELETE 。 这是真 的吗？ 

很少有刮見器支持 PUT 和 
DELETE ^ 所以为了让这些方法 
能够工作. Rails 添加了另一个名 
为 “_ methcui ” 的隐藏域来存谜 HTTP 
动作的 名字： 如果收到的请求带有 
_ method =** PUT \ Rails 将把它作为 
PUT 请求.甚至在它实际使用 POST 
提交的情，兄下也是如此， 


问‘ 


但什么是 REST 风格的设计 


它是一种设计 Web 应用的 
方式，它试图更加忠实于 web 的 
摩有设计 # 你可以在 http :// Uny . url . 
com /28 nguu 找到史多与它有关的信 
息。 
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込定一吻 CC 符 



赴时候放 iq 尔手 •人的 事件来仔细打 t 这个佐用了。加人史 
多代细的信息，更多的 m 怍，史多的视觉特效。这儿有 • 

社七相 y :. 

把 d 个:其将式左用耷 
另一个 web o..ofe. jf) ；S 
将起来如 何;' 鈞何么 
戧们不敍 

峰相括 ( TwLtter ) 3 r 


t^ 

53 -料的… 
凍年寿卿 /AI 
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你的 Rails 工異箱中的工爯 


你已经把第9章收入囊中了，现在你已经把添 
加更先进的 Rails 功能到你的 Web 应用里的能 
力加入了你的工具箱。 


\ / — Iai •!* 免、 


% 


W 一 < w ^ K fc J 

钱器4路技 〆 辑食达拓一 

, rf ^ uie - 一 
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10 真实世界里的应用 


I 真实世界 S 的 Rails + 



你已经学到了很多 Ruby on Rails 的知识。但坫为 f 能够把你的知 W 运用到真 
实世界1，你还 S 要思考儿什•杯怡。你奸该如 M 把 ㈨ 的 A 币迮抟到另一个数据库？ 
你 浚知制 测试 Rails 应用？ 你如能以•竹效地使川 Rails 和 Ruby 语言？ ft : 可以 ft 哪 
儿找到 Rails 的最新进展？请继续读 K 去,我们会 Lt 你占领有利地形并由此把你的 
开发技能提升到更商的层次。 


这是新 的一章 401 



学无止境 


我； 秦 楚我们 B 羟 揸馘到 5 Rails Web 
S 两所 t 的犬 tl 叇知 (S 是达仪仪 
是糸本知识。炙实世界1会怎祥哝 7 你不 
玎能咅诉我当伶在开外开 龙鰣所 笮的摹惰 
还体然佴夺辜中说的邡样…… 



迄今为止所有你学到的技术都是有用的，但还有更多 
需要学习的知识。 

是的，完全正确。在真实世界里发生的事情汴不总按 
照它们被设想的方式来运作。不仅如此，而 ERails 足 
一个相当庞大的框架。这本书以及任何其他少于亿万沉 
的朽都不可能完全覆盖所有的知 U! 点。 

但没有任何理由来认为你还没有准备 W ! 请浏览这敁 
后的十几页来看看你掌捤 f 什么，并顒便学几招你还 
不知逬的技巧。 
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我们 0 U 乂介绍 fRails 大 t 辅助函数中的很小一部分，而♦实上还有 
更多可以选择的辅助函数。请符一下你是否能够把 F 面的辅助函数 
匹配到它们的实际用途。 


number 一 to 一 phone 

il : 浏览器 fi 动探测•个 RSS 源。 

number to percentage 

允许你为校板代码的运行时 Nit •时。 

error_message_on 

把数字格式化成美国电话号码。 

auto 一 discovery 一 link_tag 

把数卞格式化成百分数。 

image 一 tag 

返 N 年、月、日的选择 （ select ) 标 
签。 

benchmark 

帑助你格式化错误消息 0 

date_select 

返回你可以在模板里使用的阁像 
( image ) 标签。 
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了解你的辅助函数 


: 




寡 


我们仅仅介绍了 Rails 大景辅助函数中的很小一部分，而事实上还有 
更多可以选择的辅助函数。请看一下你是否能够把下面的辎助函数 
匹配到它们的实际用途。 



( image ) 标签。 


扪笮 紅妫 - 

想要査看更多关于 Rails 气前版本可用的辅助函数的话， 

请至 http://tinyurl.com/railshelpers, 
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着 r 迖儿布满满一页的 Ruby “试试着” 

你是否注意到你只需7■解很少的 Ruby 知便能开发很酷的 Rails Web 应 
用？即使如此，多了解一些 Ruby 有时也会很有用。这儿有一些你可能 
想要尝试的 Ruby 例子，请输人它们宋看看会发生什么…… 


读取文件的所有行到一个名为 “a” 的数组中 

创建一个带有50个=符号的字符串 

a=File. readlines ("filename.txt") 

"="*50 

对这个数组进行排序 

从字符串中获取单词数组 

a.sort 

"tD be omottD be "專 比 

把字符串《转过来 

返回对象的类（数据类型） 

"Bassackwaris"jBveise i 

or:^iss 

返 M 翻转过的字符串拷贝 

对浮点数取整到最接近该浮点数 

"BasBackwajds"jEvei9e 

(3 14) jound 

字符串 s 是否包含 “ Waldo ” ？ 

求平方根 

M aHo/s 

M ath J93rt0.6) 

字符串是一个邮政编码吗？ 

删除文件 

r\d{5}$/=- -90210- 

File de fete ("Cifenam e isct") 

把字符串转化成整数 （Fixnum) 

当前 n 期和时间 

-I2345"lD_i 

Tin e jiow 

把卞符串转化成浮点数 

当前年份 

-31415 - JD_f 

Tin enow year 

把对象转化成字符串 

给方法取个别名 

a Jd_s 

a iBsanothetjiam e m y—m ethod 

美化打 印数组 的内容 

返回 H 录中的文件数组 

fL,2,3,4,5j 

D icentXBsCtSiBctDiyNam e") 

美化打印哈希的内容 


{a=>l, i)=>t ■} ji^>ect 



>试试弟7 

C + 
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测试 


Web 应用也 f 要测试 

自动化测试是软件开发中至关重要的一环，而 ft 到现在我们都还没 
有说到它。为什么呢？测试软件的代码需要你彻底理解你所使用的工 
具，而设计测试用例则比写代码本身更加困难（也更加冇 趣）。 这就 
是为什么本书着價 给予 你理解 Rails 如何工作和思考的技能。只有你理 
解之后你才可以开始考虑如何测试应用。 

但这并不意味肴你要在系统完成很久之后才开始测试。根本不是这样 
的。 ft 好的测试用例应该在你编写你的主程序之前就12经写好。 

A 

Rails 本身提供了很多测试支持，几乎比其他所有的框架都多。毎个佐 
用都位含了一 绀测试 脚本（在 testH 录 下）， 而毎次你生成支架程序 
时， Rails 也会为你生成一组标准测试用例。所以，如果你来到第 1 章 
中编3 i 架式 i 了票系统所在的目录 汴 R 输人： 





Rails 将为你运行整个测试集。这是否意味着你不再需要编写你自己 
的测试用例呢？拳实上，答案足否定的。作为一名 Rails 开发人员， 
你的大部分时间将会花在编写和维护测试用例上。 
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布啷些类型的测试玎用喂？ 

有三种主要的 测试： 

$无谢试 

Rails 柯时使用-些不同干其他地方的术语。在大多数系统里，“单龙测试 
(unit test ) M 是指任 H 独立代码段的测试。 ffiRails 的定义要更 ft 体一些。在 
Rails 中，“中.元测试”垃指模型类的测试。毎当你 S 接或者通过支架间接生 
成模型， Rails 都会为你在 test / unitU 录下创违标准的单元测试用例 

功 能谢试 

Rails 的功能测 ut 是指独么控制器的测忒。功能测试检作你足和 发送了 特定的 
请求，你是否获得了特定的响应。你町以在 test / functbnalll 录 电找到 
功能测甙用例。同样，每当你 ft 接或*通过支架间接生成控制器时， Rails 会 
创建 功能测 忒用例。 

鑲 成测试 

集成测试是高层次的测试，它#起来有些像测 lit 人员使用的测试脚本。集 
成测试用例从整体上测忒系统。所以它们自动化了典型用户在你的系统上可 
能做的各种动作。银成测试用例冇个专门的 tJ 录 （ test / jitegratbn ) , 
但是它们并不会 G 动牛成。它们需要针对你的系统来特别设计，所以你需要 
自行创逮它们。 

最后，所有测忒用例的测试数据都保存在 test / fixtures 下的数据文件中。 
fixtureR 是测试数据集的一个花哨的名字而已。 Rails 将把 fixtures 中的数据存 
储在一个特別的、独立的测试数据库中来使得你的开发（或者实时 数据） 不 
会与你的测试所需数据混 m 起来。 

请査看 http://linyurl.com/railstest 来获取吏多关于 Rails 中的测试的信息。 



部署你的应用 

上线运行 

你的应用不会一直停留在开发阶段，在特定时候，你就可以把它上线了。 
然后你该怎么做呢？在你的应用代码中指定数据库位置或者请如此类的配 
置信息不是一个好主意。毕竟，你并不希望运行版本和测试版本的代码有 
着不同的实现。你只是希望它们使用不同的数据库而已。 

这就是为什么 Rails 让你指定环境 （ environment ) 。环境确定了数据库的 
位置和类型以及其他一些设 S ，比如应该记录多长的日志消息。 

默认情况下，一个应用被设定使用三种不间的 环境： 


o 开发环境 

这是默认使用的环境，也是本书一直使用的环境。开发环境使用 db / 
development . sqlite 3 数据库。 

o 测试环境 

这个环境被设罝用干自动测 w 脚本。 

o 产品环境 

这是你的上线 dive ) 环境。 


你淡如何在不罔环煥之问換喂？ 


当你启动服务器时， Rails 査找一个名为 RAI£_ENV 的环境变1;。这 
将告诉 Rails 使用哪个环境来运行。如果你希望从开发环境切换到产品 

环境，你就需要设置 RALS 一 ENV 变量： 如羃焓使用你靠 f 


> set RAILS 一 ENV=production 

> ruby script/server 



> RAI LS __ ENV=p rodu c ti on 

> ruby script/server 
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真实世界里的应用 


那么你淡如何 E 改数椐库£1哝? 


如果你査看 con fig/data base .ym 玫件，你就会发现毎种环境 
下的数据库配1信息。 

例如，你原来的 SQLite 产品环境可能设置如 下： 



但是如果你想更改产品环境来使用 Oracle 数据库，那么它可能会被 
改成如下的 配罝： 


development : 

adapter: oracle 
database : mydatabasename 
username: scott 
password: tiger 


或者，如果你希望上线环境使用安装在与 Rails 相同的机器 JL 的 
MySQL 数据库，你可以把配燹 改成： 


@1 
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好好了解 REST 


什么是 REST ? 

我们已经在本书中多次看到过 REST 。 Rails 如何 使用 REST ? REST 设 I 十如 
H 成为新的 Rails 指导原則？如果你使用 REST , 你的牙&会更加闪闪发 
光，你的生活会更加偷快，而整个世界都会变得美好和灿烂。 

让我们从基本槪念着手。 REST 是指表述性状态转变 (Representational 
State Transfer 的缩写），它是一种人们如何设计计算机系统的方式。很 
显然，我们周围最重要的计算机系统就是万维 M (World Wide Web ) , 
特别值得注意的是提出 REST 的人 —— Roy Fielding ——也是 HTTP 蚬范的 
作者之一。 


为什么 HTTP 作者同时也是 REST 作者就显得重要呢？好吧，因为 REST 风 
格的设计就意 味翁你 的应用将按照 Web 最初的设想来运作。 

邡么什么是 RKT 的主要 设计准 則？ 

O 最重要的是资源 （ resource ) 。 

这意味着你的系统中所冇重要数据郎是你可以处理的、独立的 、 《r 
识别的事物。如果你有一个卖甜甜圈的网站，那么甜甜圈就迠资源。 

o 每个资源都有一个合适的名字。 

在 Web 上，这就意味着每件东西都有一个 URL 。 

o 你可以对资源执行一组标准操作。 

CRUD ( Create 、 Read 、 Update 、 Delete ) 操作是很典切的操作集合, 
它们被 Rails 和 Web 所支持。 

o 客户端和服务器彼此无状态地进行对话 

这就意味着当一个客户端（比如浏览器）与 REST 风格的应用对话 
时，这个过程是一组完全孤立的请求和响应。客户端向服务器发送 
请求，服务器返回响应，然后对话结束。 


所有这些#起來都显而易见，+是叫？它们非常好地描述了 Web 是 
如何工作的。 

它们确实曾经是 Web 如何工作的很好的描述。在 Web 变糟糕之前…… 
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真实世界里的应用 


迷失方向的 Web 庋阁 

设想一下有个 Web 应用允许人1販卖多余的火箭零部件： 

开发人员可能会创违-个如下的显■示 火箭零 部件的系统： 

http://vTOW.boosters-r-us.com/airframes/472 

这个网站与火箭零部件有关，而这个 URL 可以被甩来作为该部件的名 
字。 

但看 ft 当有人吏新这个部忭的信息时会发生什么，比如说它的 价格。 

系统里的 Web 表 单把信 息提交到这个 URL : 

http://www.boosters-r-us.com/airframes/472/update 

这种做法的问题迮于它没有遵媚 REST 原奶 L 为什么这么说呢？因为在 
RESTW •格的系统里， URL 应该是资源名=而这儿的第二个 URL 并不表 
示一件事物，它表示了一个动作。 

为什么不迸循 REST 原则会布问涯嘴？ 

你是否曾经在浏览器甩車新访问一个 URL 并被洵问是否希望再次发送 
数据”浏览器历史仅仅是 URL 的列表而这意味着它同时也应该是名字 
的列表。佴是如果一个 Webft 用使用 URL 来表征活动，那么当你通过 
你的历•史记录来后退时，浏览器将无法知道你是否㈣ -再做■次该动 
怍。 

你 Rf 要使用 HTTP 动询 

我们该如 M 解决 这个问题呢？ REST 的第三准則表明需 要有一 组定^•好 
的动作。 RESTML 格的应用使用 HTTP 方法来定义活动且让 URL 表示资 

源的 名字： f C 甚友孕式在用中的 R & STM 格珞 

由 f 的 


CRUD operation 

HTTP method 

URL 


Create a component 

POST 

http://www.boosters-r-us.com/airframes/ 

Read a component 

GET 

http://www.boosters-r-us.com/airframes/472 

Update a component 

PUT 

http://www.boosters-r-us.com/airframes/472 

Delete a component 

DELETE 

http://www.boosters-r-us.com/airframes/472 

1 
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是的，还有更多的知识 


生活在 Edge 上 

Rails 直 都在变化中，你如何才能跟 t 不断增加的最新最好的特性呢？ 一 
种办法是运行在 Edge 之上。 

Rails 通过下栽并直接安装最新版本的框架到你的应用程序来让你4以轻易 
便运行在最近的 Rails 版本（被称为 Edge Rails ) 上。 

现在一些其他应用框架想要更新到最新框架版本会异常困难。你不得不通 
过 Web 浏览器来下载文件，阅读最新的安装指南，设定路径，确保配 S 匹 
Sd 你的系统，以及诸如此类的步骤。这个过程非常复杂以至干很少有人愿 
意这么做。 

但是很多人都在使用 Edge Rails 。 为什么呢？好吧，这不仅仅足因为他们希 
琪使用最新特性 。 Rails —《都在高速的发展过程中，你可能会发现即使足 
很小的一次升级都可能比你的应用的一部分代码失效。所以为了确保他们 
的应用能够始终跟 hRails 的演进，他们不希望等 k 儿周或者几个月的时间 
朿升 级，有些人 Jt •至毎天都会更新•次 Edge 版本。 

俏是你如何才能安装 Edge Rails 到你的应用中呢？很简单，你只需要这样 
做： 



这条命令就是你所要的一切。 rake 工具会连接到 Rails 开发服务器，下载 
Rails 脚本的最新版+，然后把它们安装到你的应用的 vendor / railsH 录中。 
每次当你启动 Rails 服务器时， verndorH 录中的代码会在本机的七 Rails 程序 
运行之前生效。这就意味着 Edge Rails 将会被用？这个应用。 

在 Edge 版本上的生活可能会相当惊险。但是有时最好找出版本兼容性的 
问题，一次一个 . 
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获取 E 多的信息 

即使 Rails 允 I 午你快速且可靠地创建全功能的 Web 应用，但毋庸1疑的是你 
需要很长的时间才能够真正掌捤 Rails 。 因为有太多的知识要学。 

这就意味着你需要一本非常不错的参考手册。而蜋好的手册都在网 
上， Rails —直都在变化中。每天都有新的功能添加到 Rails 源代码屮，而 
唯-与它保持同步的方法就是上网。这儿有一些很不错的网站可以让你上 
手： 

O http :// www . rubyonrails . org / 

Rails 的主页。它不仅仅包含软件，也包含演示、视频和进一步阅读 
的链接。 

http :// wiki . rubyonrails . org/rails 

这里提供了安装的详细指南和故障诊断，也提供了更多在线资源的 
链接。 

http ：// ryandaigle . com / 

Ryan 的博客包含了大量的你可以在 Rails 中施展的最新招数。 
http :// www . ruby - lang . org / en / 

Ruby 语言的最新进展。 

沟 S 文裆 


除了大盪的在线资料，你的 Ruby on Rails 也包含了你所需要的大部分 
内容，它们可以直接使用。最重要的两个命令行工 具是： 
rioom ethhg> 


这里的 < something >^ 你想要了解的 Ruby 类。例如 ， “ri Array ” 将会 
告诉你关于 Array 类的一切。 

另一个有用佶息的来源是 gem 服务器。 Gem 是最常用的 Ruby 包管理工 
具，而且它可能也是你用来安装 Rails 的命令。 Gem 有个内 S 的服务器， 
它可以提供与你在 h 11 p ://a p i. ru by on ra i 1 s. org 网站上能够找到的一样的 
API 文档。你可以输人如下命令来启 动它： 


gem server 


然后在如 KURL 进行 浏览: 


hop ^/bca hostfl 808 / 



睡前对读 


洎遣性读物 . 

当然，这儿是 Head First Labs , 我们垃作者。无论在线资料多 
么美好，它们也比不上一本真实的町以帮助你理解知识的书。现在你 
已经来 到了本 书的结部分，你的头貼觉得很舒服而 II 允满了新的 
Ruby on Rails 专业知识，你可能想要抓住这个机会来试试 Jt •他一些相 
关书籍。 



«The Ruby Way 》 

我们喜爱这本“深入浅出”系列的书。它是一本内容丰富，但垃又非 
常优美的茗作，由 Hal Fuhon 撰写。这本朽将带你进行一次 Rubyig 言 
的深人之旅。尤其1要的迠，它不仅告诉你语 a 的细节，而且它还解 
释了 ft 语#设的哲学。 Rails 如此欹越的很多因尜部沉接東•肉干 
Ruby , iWJt •中的许多 L 相尜都在本书屮釘所提及。 


《Agile Web development with Rails)) 

这迫一本很捧的 15, 它会把你带人 Rails 开发。它的有趣之处在于它 
像一个开发项目 * 样来撰所以几个月前新版刚_发布， beta 版本 
就被放/£网上以供人们阅读和 评论。 



Rails 

Cookbook 



((Rails Cookbook 》 

• H - 你准备好使用 Rails , 你就町能盂要解决很多很多其他人在你之前 
W 经不得不处理的问题。不要宙怕 ！ 《Rails Cookbook 》 会给你一组美 
味的预先写好的代码来帮助你度过难关。 
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真实世界里的应用 


相兵活翅的“深入浅出”系列书籍 

现在，除了 Ruby 和 Rails 方面的书锫，你可能会发现研读一些相关话 
题的书賭也有帮助。什么是引导你的大脑进人一个新课题的最佳手段 
呢？当然是通过“深人浅出”系列书耪！ 



〈〈深入浅出 Ajax (Head First Ajax) > 

Rails 提供了大 M 内 1 的 Ajax 开发支持，但娃为了吏好地使用它们，你 
需要/解 Ajax 是如 H 工作的。还有什么比《深人浅出 Ajax 》 更好的方法 
吗？ 


〈〈深入浅出 JavaScript (Head First JavaScript) » 

Ajax 54 立在 JavaScript 之上，如果你作常 广解如 M 使出 JavaScript , 你 
将吋以让你的应用变得吏加 强大。 《深人浅出 JavaScript 》 足进人该语言 
的很好的方法。 



Head First 

Software 



〈〈深入浅出软件孖龙 (Head First Software 
PmlopmeHt) 》 

在本书中，你已经学会了如何在 Ruby on Rails 框架里开发。如果你希笾 
从编程进步到开发，那么就选择这本书吧。它将教你如何在你的项 H 屮 
执行计划，如何&动化测试和持续集成测 W 。 
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rails 工具箱 


綾 


你的 Rails 工與箱中的工爯 


P 你已经把第10章收入囊中了，现在你已经把 
你黑要思考的几件真实世界里的事情加入了你 
的工具箱。 



■ r ^ ILs X S 

咖_含- 大推糾 以用衫用 ？ 的錄外 的样 ㈣ 

銮個 祕蛣‘ 比应 用的， 3斜曼多 >y ’。 ] 

⑽你 t—M • 從 M 糾 t) _融 
㈣ raiU ： fr £C z^—^^ ㈣ 祕 
你的在用 1 

RAics_eNV=-prod^tw.^ -祕⑽如巧 * 一 

个“在士”數搞庳^ # ^^ 

>——给你笑子推 ㈣ 叫的象的方.在 


—4 幼 i^by 之枯掩务器 
ii 4你•捅有的蚤発丈的幵发 2 
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离孖 Rails 村 


真实世界里的应用 



很窩兴 有你在 Rails 村 

看到你的离去我们很伤心， 么比 •〒: 以致 mai 的 r。 你的 Raii s 
之旅I刚刚开始，我们 Li 经把你放在驾驶座I:了。我们非常想知逬你的进展，所 
以请在 Head First Labs 网站 ( www.headfirstlabs.com ) 上给我们-点反饿，让我 ff】 
知道 Rails 是否真的 ih 你的人生变得多髟！ 
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定做电子书，海量电子书， 
各科电子书，代寻电子书。 


Q Q ： 1759560190 
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